众所周知,Golang 的编译速度是非常之快的。在设计 Go 时,编译速度是一个重要的考虑因素。但是,你是否关注过 Go 编译代码后生成的二进制可执行文件的大小?让我们来看一个简单的 HTTP 服务的应用示例:
import (
"fmt"
"net/http"
)
func main() {
// create a http server and create a handler hello, return hello world
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
fmt.Fprintf(w, "Hello, World\n")
})
// listen to port 8080
err := http.ListenAndServe(":8080", nil)
if err != nil {
return
}
}
编译后的大小为 6.5M。
➜ binary-size git:(main) ✗ go build -o server main.go
➜ binary-size git:(main) ✗ ls -lh server
-rwxr-xr-x 1 staff 6.5M Jul 2 14:20 server
现在,让我们尝试优化上面应用的大小。
默认情况下,Go 编译器会在编译后的程序中包含符号表和调试信息。通常情况下,你可以在发布版本中移除这些调试信息,以减小二进制文件的大小。
➜ binary-size git:(main) ✗ go build -ldflags="-s -w" -o server main.go
➜ binary-size git:(main) ✗ ls -lh server
-rwxr-xr-x 1 hxzhouh staff 4.5M Jul 2 14:30 server
UPX^[1]^ is an advanced executable file compressor. UPX will typically reduce the file size of programs and DLLs by around 50%-70%, thus reducing disk space, network load times, download times, and other distribution and storage costs.
UPX 是一种先进的可执行文件压缩器。UPX 通常能将程序和 DLL 文件的大小减少 50%-70%,从而减少磁盘空间、网络加载时间、下载时间以及其他分发和存储成本。
在 Mac 上,你可以通过 brew 安装 UPX。
brew install upx
UPX 有很多参数,其中最重要的是压缩率,范围从 1 到 13。让我们看看仅使用 UPX 就能将二进制文件大小缩小多少。
➜ binary-size git:(main) ✗ go build -o server main.go && upx --brute server && ls -lh server
-rwxr-xr-x 1 hxzhouh staff 3.9M Jul 2 14:38 server
6.5M-> 3.9M 压缩比约为 60%。
同时启用 UPX 和 -ldflags="-s -w":
➜ binary-size git:(main) ✗ go build -ldflags="-s -w" -o server main.go && upx --brute server && ls -lh server
-rwxr-xr-x 1 hxzhouh staff 1.4M Jul 2 14:40 server
最后,我们得到的可执行文件大小为 1.4M,而未经压缩的文件大小为 6.5M,节省了大约 80% 的空间。这对于大型应用程序来说是相当可观的。
压缩后的程序与原始程序一样,无需解压缩即可正常运行。这种压缩方法称为外壳压缩,包括两个部分:
执行时还包括两个部分:
如果对编译大小没有严格要求,可以选择不使用 UPX 压缩。一般来说,服务器端独立后台服务不需要压缩大小。不过,如果是在 Raspberry Pi 等嵌入式设备上,还是值得一试。
这个帖子有很多很有意思的答案:
参考资料
[1]UPX: https://upx.github.io/
[2]how-to-reduce-go-compileed-file-size: https://stackoverflow.com/questions/3861634/how-to-reduce-go-compiled-file-size
All comments