- Published on
关于 Go 的编译及体积优化
- Authors
- Name
- Darwin Stone
- @dustred233
编译方式
静态编译
在Go中,可以使用静态编译来将Go程序编译为一个独立的可执行文件,其中包含了所有的依赖库和运行时环境,不需要依赖外部的共享库。这样就可以在没有Go语言环境的机器上直接运行编译后的可执行文件。
# output-file 是输出的可执行文件名
go build -o output-file main.go
优点是可以将这个可执行文件复制到其他机器上,无需安装Go语言环境,就可以直接运行它。
缺点是生成的可执行二进制文件的体积较大。
动态编译
Go语言本身不支持动态编译(即在运行时动态加载和执行代码)。
但是 Go 1.8及以上版本引入了插件机制,允许在运行时动态加载和执行插件代码。
可以将插件代码编译为插件模块,然后在主程序中通过插件机制加载和执行该模块。使用plugin包可以实现这个功能。
另外,可以使用Go语言的外部库来实现动态加载和执行其他语言的代码。
例如,使用cgo库可以与C语言进行交互,通过加载和执行C代码来实现动态编译的效果。也可以使用与其他语言的交互库,如go-python、go-lua等,来加载和执行相应语言的代码。
减少静态编译后的二进制体积
1. 指定编译参数
# -s 去掉符号信息
# -w 去掉调试信息
go build -ldflags="-s -w" -o output-file main.go
实测可以原本的 33M 可以减少到 27M。
2. 压缩
UPX https://github.com/upx/upx 是一个可执行文件的压缩工具。
UPX压缩包含两个部分:
- 在程序开头或其他合适的地方插入解压代码;
- 将程序的其他部分压缩。
程序执行时,也包含两个部分:
- 首先执行的是程序开头的插入的解压代码,将原来的程序在内存中解压出来;
- 再执行解压后的程序。
也就是说,upx 在程序执行时,会有额外的解压动作,不过这个耗时几乎可以忽略。
如果对编译后的体积没什么要求的情况下,可以不使用 upx 来压缩。一般在服务器端独立运行的后台服务,无需压缩体积。
# 安装UPX
wget https://github.com/upx/upx/releases/download/v4.0.2/upx-4.0.2-amd64_linux.tar.xz
tar -Jxf upx*.tar.xz
sudo cp upx*/upx /usr/bin
upx -v
# 压缩
upx [options] yourfile
-1[23456789]:不同的压缩级别,数值越高压缩率越高,但耗时更长。
对于小于 512 KiB 的文件默认使用 -8,其他的默认为 -7。
--best:最高压缩级别
--brute:尝试使用各种压缩方式来获取最高压缩比
--ultra-brute:尝试使用更多的参数来获取更高的压缩比
-o [file]:将压缩文件保存为 [file]
优点
UPX 可以压缩各种类型的可执行文件
压缩后的文件可以直接由操作系统执行
压缩过程不会修改源文件,也就意味着解压后直接可以得到原始文件
不会产生额外的动态库调用
缺点
- 运行的程序不会共享数据段(汇编),所以多实例运行的程序不适合压缩
- 使用 ldd 和 size 命令无法获取到程序的有效信息
# 具体使用
go build -ldflags="-s -w" -o output-file main.go && upx output-file
实测可以将 33M 减少到 7.1M 。