是否可以通过使用 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