ℹ️ 本文基于 Go 1.13。

符号表是由编译器生成和维护的,保存了与程序相关的信息,如函数和全局变量。理解符号表能帮助我们更好地与之交互和利用它。

符号表

Go 编译的所有二进制文件默认内嵌了符号表。我们来举一个例子并研究它。下面是代码:

var AppVersion string

func main() {
 fmt.Println(`Version: `+AppVersion)
}
nm
0000000001177220 b io.ErrUnexpectedEOF
[...]
0000000001177250 b main.AppVersion
00000000010994c0 t main.main
[...]
0000000001170b00 d runtime.buildVersion
bAppVersionbdt
nmgo tool nm
1177220 B io.ErrUnexpectedEOF
[...]
1177250 B main.AppVersion
10994c0 T main.main
[...]
1170b00 D runtime.buildVersion

当我们知道了暴露的变量的名字后,我们就可以与之交互。

自定义变量

go build
-X-X
go build -o ex -ldflags="-X main.AppVersion=v1.0.0"

构建并运行程序,现在会展示在命令行中定义的版本:

Version: v1.0.0
nm
1170a90 D main.AppVersion
bdstring
D runtime.badsystemstackMsg
D runtime.badmorestackgsignalMsg
D runtime.badmorestackg0Msg
B os.executablePath
B os.initCwd
B syscall.freebsdConfArch
D runtime/internal/sys.DefaultGoroot
B runtime.modinfo
B main.AppVersion
D runtime.buildVersion
DefaultGoroot

调试

符号表的存在是为了确保标识符在使用之前已被声明。这意味着当程序被构建后,它就不再需要这个表了。然而,默认情况下符号表是被嵌入到了 Go 的二进制文件以便调试。我们先来理解如何利用它,之后再来看怎么把它从二进制文件中删除。

gdbgdb exlist
GNU gdb (GDB) 8.3.1
[...]
Reading symbols from ex...
Loading Go Runtime support.
(gdb) list 10
6
7  var AppVersion string
8
9  func main() {
10    fmt.Println(`Version: `+AppVersion)
11 }
12
(gdb)
gdb-ldflags=-s
GNU gdb (GDB) 8.3.1
[...]
Reading symbols from ex...
(No debugging symbols found in ex)
(gdb) list
No symbol table is loaded.  Use the "file" command.
-s[DWARF](https://golang.org/pkg/debug/dwarf/ "DWARF")

二进制文件的大小

去掉符号表后会让调试器用起来很困难,但是会减少二进制文件的大小。下面是有无符号表的二进制文件的区别:

2,0M  7 f é v 15:59 ex
1,5M  7 f é v 15:22 ex-s
cmd/go
14M  7 f é v 16:58 go
11M  7 f é v 16:58 go-s

这里没有符号表和 DWARF 信息,也小了 25%。

如果你想了解为什么二进制文件会变小,我推荐你阅读 WebKit 团队的 Benjamin Poulain的文章“不寻常的加速:二进制文件大小”。

链接:https://medium.com/a-journey-with-go/go-how-to-take-advantage-of-the-symbols-table-360dd52269e5

本文链接:http://www.yunweipai.com/42696.html