Go死码消除

概念:

死码消除(dead code elimination, DCE) 是一种编译器优化技术, 作用是在编译阶段去掉对程序运行结果没有任何影响的代码

和 逃逸分析[1],内联优化[2]并称为 Go编译器执行的三个重要优化


效果:

对于 const.go代码如下:

复制代码
package main

import "fmt"

func max(a, b int) int {
 if a > b {
  return a
 }
 return b
}

const a, b = 10, 20

func main() {
 if max(a, b) == a {
  fmt.Println(a)
 }
}

对于var.go代码如下:

复制代码
package main

import "fmt"

func max2(x, y int) int {
 if x > y {
  return x
 }
 return y
}

var x, y = 10, 20

func main() {
 if max2(x, y) == x {
  fmt.Println(x)
 }
}

两个文件的差异,只在于 两个参数 是变量还是常量

分别编译 const.govar.go, 生成的二进制文件大小如下:

go build -o 想要生成的二进制名称 想要编译的.go文件

不难发现, constvar 体积小了约 10%

为何如此?

首先编译器会对max函数 进行内联优化, const.go 优化后如下:

复制代码
package main

import "fmt"

const a, b = 10, 20

func main() {
 var result int
 if a > b {
  result = a
 } else {
  result = b
 }
 if result == a {
  fmt.Println(a)
 }
}

因为 ab 是常量, 永远有a<b, 编译器可以在编译时证明该分支永远不会为true, 因此编译器可以进一步优化代码为:

if a > b {}这个分支被消除了,这称为分支消除

又知道结果总是等于b,因此编译器还将进一步将代码优化为:

复制代码
package main

const a, b = 10, 20

func main() {
 const result = b

}

最后就是:

复制代码
package main

func main() {
}

而对于var.go, 参数为 全局变量 不为常量,编译器并不知道运行过程中x、y会不会发生改变, 因此不能进行死代码消除.

这部分代码被编译到最终的二进制程序中, 造成 二进制文件 varconst 体积大了约 10%

分支消除是死码消除一种. 使用静态证明来表明一段代码永远不可达,通常会被称为死代码,它不需要在最终的二进制文件中编译和优化.

编译器在编译阶段, 死码消除与内联优化一起工作, 可以减少循环和分支产生的代码数量

参考资料

1

逃逸分析: https://dashen.tech/2021/05/29/golang逃逸技术分析/

2

内联优化: https://dashen.tech/2021/05/22/Go中的内联优化

本文由mdnice多平台发布

相关推荐
落枫5919 小时前
OncePerRequestFilter
后端
程序员西西19 小时前
详细介绍Spring Boot中用到的JSON序列化技术?
java·后端
课程xingkeit与top19 小时前
大数据硬核技能进阶:Spark3实战智能物业运营系统(完结)
后端
课程xingkeit与top19 小时前
基于C++从0到1手写Linux高性能网络编程框架(超清)
后端
语落心生19 小时前
探秘新一代向量存储格式Lance-format (二十二) 表达式与投影
后端
码事漫谈20 小时前
音域之舞-基于Rokid CXR-M SDK的AI眼镜沉浸式K歌评分系统开发全解析
后端
上进小菜猪20 小时前
基于 Rokid CXR-S SDK 的智能提词器开发全解析——AI 应答辅助系统
后端
Rust语言中文社区20 小时前
【Rust日报】 丰田“先锋”选择了 Rust
开发语言·后端·rust
椎49520 小时前
苍穹外卖资源点整理+个人错误解析-Day10-订单状态定时处理(Spring Task)、来单提醒和客户催单
java·后端·spring
努力的小雨20 小时前
从零跑起 RokidDemo:开发小白也能搞定的入门实践
后端