是否可以通过使用 TinyGo 编译 Go 二进制文件来使其更小?
Is it possible to make a Go binary smaller by compiling it with TinyGo?
TinyGo Homepage 说它是为微控制器(或MCU)设计的。我想知道是否可以使用 TinyGo 将 go 编译成 ARM/Linux 系统的小型二进制文件?
目前UPX --best --lzma
,我的go代码编译成arm binary大约1MB,我希望能再小一点,因为系统内存有限
tl;dr:基本上是的,但您可能需要移植您的程序。
I wonder if it is possible to use TinyGo to compile go into small binary for ARM/Linux system?
是(参考:https://tinygo.org/docs/guides/linux/)
TinyGo also lets you compile programs for Linux systems, both 32-bit and 64-bit, on both x86 and ARM architectures.
For cross compiling, you can use GOOS and GOARCH as usual.
例如:
GOARCH=arm GOOS=linux tinygo build -o example ./...
至于与官方 go
compiler/runtime 相比,可执行文件的体积有多大,这可能取决于您的程序的具体情况。的确,TinyGo 存在的理由 是为了提供更小的二进制文件。
因此,我们可以尝试构建一些可执行文件并根据经验比较大小。顺便说一句,我意识到这种比较在 TinyGo overview page 中可用。无论如何,我们可以在这里重现它,并提供一些额外的细节:
首先,Go 1.17 尚不支持 (GitHub issue)。您必须在系统中安装 Go 1.16 或更低版本才能构建或使用 TinyGo dev
分支。
然后,正如我在评论中提到的,请记住 TinyGo 不支持编译所有标准库包,因此如果您的代码使用其中一个,则必须移植它。这是支持和不支持的软件包列表:https://tinygo.org/docs/reference/lang-support/stdlib/
为了运行比较,我下载了Go 1.16.5,TinyGo今天支持。
# reference: https://golang.org/doc/manage-install
$ go install golang.org/dl/go1.16.5@latest
$ go1.16.5 download
程序:
package main
import (
"fmt"
)
func main() {
fmt.Println("Hello World!")
}
我在 mac,所以我必须使用适当的环境变量进行构建。我按照评论中的建议使用 -ldflags="-s -w"
进行构建。
$ go env | grep -E "(GOOS|GOARCH)"
GOARCH="amd64"
GOOS="darwin"
$ go1.16.5 build -o example_go ./...
$ go1.16.5 build -ldflags="-s -w" -o example_go_ldflags ./...
$ GOROOT="/Users/blackgreen/sdk/go1.16.5" GOARCH=amd64 GOOS=darwin tinygo build -o example_tinygo ./...
$ du -hs example_*
1.9M example_go
1.5M example_go_ldflags
64K example_tinygo
对于一个简单的 hello world 程序,有一个显着的收获。不出所料,它也明显优于 ldflags
。
TinyGo Homepage 说它是为微控制器(或MCU)设计的。我想知道是否可以使用 TinyGo 将 go 编译成 ARM/Linux 系统的小型二进制文件?
目前UPX --best --lzma
,我的go代码编译成arm binary大约1MB,我希望能再小一点,因为系统内存有限
tl;dr:基本上是的,但您可能需要移植您的程序。
I wonder if it is possible to use TinyGo to compile go into small binary for ARM/Linux system?
是(参考:https://tinygo.org/docs/guides/linux/)
TinyGo also lets you compile programs for Linux systems, both 32-bit and 64-bit, on both x86 and ARM architectures.
For cross compiling, you can use GOOS and GOARCH as usual.
例如:
GOARCH=arm GOOS=linux tinygo build -o example ./...
至于与官方 go
compiler/runtime 相比,可执行文件的体积有多大,这可能取决于您的程序的具体情况。的确,TinyGo 存在的理由 是为了提供更小的二进制文件。
因此,我们可以尝试构建一些可执行文件并根据经验比较大小。顺便说一句,我意识到这种比较在 TinyGo overview page 中可用。无论如何,我们可以在这里重现它,并提供一些额外的细节:
首先,Go 1.17 尚不支持 (GitHub issue)。您必须在系统中安装 Go 1.16 或更低版本才能构建或使用 TinyGo dev
分支。
然后,正如我在评论中提到的,请记住 TinyGo 不支持编译所有标准库包,因此如果您的代码使用其中一个,则必须移植它。这是支持和不支持的软件包列表:https://tinygo.org/docs/reference/lang-support/stdlib/
为了运行比较,我下载了Go 1.16.5,TinyGo今天支持。
# reference: https://golang.org/doc/manage-install
$ go install golang.org/dl/go1.16.5@latest
$ go1.16.5 download
程序:
package main
import (
"fmt"
)
func main() {
fmt.Println("Hello World!")
}
我在 mac,所以我必须使用适当的环境变量进行构建。我按照评论中的建议使用 -ldflags="-s -w"
进行构建。
$ go env | grep -E "(GOOS|GOARCH)"
GOARCH="amd64"
GOOS="darwin"
$ go1.16.5 build -o example_go ./...
$ go1.16.5 build -ldflags="-s -w" -o example_go_ldflags ./...
$ GOROOT="/Users/blackgreen/sdk/go1.16.5" GOARCH=amd64 GOOS=darwin tinygo build -o example_tinygo ./...
$ du -hs example_*
1.9M example_go
1.5M example_go_ldflags
64K example_tinygo
对于一个简单的 hello world 程序,有一个显着的收获。不出所料,它也明显优于 ldflags
。