golang在编译时用ldflags设置变量的值

我们经常会在一些程序的输出中看到程序版本、编译时间、Git的commit id等信息,比如docker

1
2
3
4
5
6
7
8
9
ming@vultr:~$ docker version
Client:
Version: 1.12.5
API version: 1.24
Go version: go1.6.4
Git commit: 7392c3b
Built: Fri Dec 16 02:42:17 2016
OS/Arch: linux/amd64
...

我们可以提供一个配置文件version.conf,程序运行时从version.conf取得这些信息进行显示。但是在部署程序时,除了二进制文件还需要额外的配置文件,不是很方便。
或者将这些信息写入代码中,这样不需要额外的version.conf,但要在每次编译时修改代码文件,也够麻烦的了。
有一种更好的办法是在编译时使用参数-ldflags -X importpath.name=value,官方解释如下

-X importpath.name=value
Set the value of the string variable in importpath named name to value.
Note that before Go 1.5 this option took two separate arguments.
Now it takes one argument split on the first = sign.

以下面代码说明

1
2
3
4
5
6
7
8
9
10
11
12
13
package main

import "fmt"

var (
VERSION string
BUILD_TIME string
GO_VERSION string
)

func main() {
fmt.Printf("%s\n%s\n%s\n", VERSION, BUILD_TIME, GO_VERSION)
}

用如下命令编译,注意因为datego version的输出有空格,所以main.BUILD_TIMEmain.GO_VERSION必须使用引号括起来

1
go build -ldflags "-X main.VERSION=1.0.0 -X 'main.BUILD_TIME=`date`' -X 'main.GO_VERSION=`go version`'"

编译成功后运行程序,结果如下

1
2
3
4
ming@ubuntu:~/go_workspace/src/test$ ./test 
1.0.0
Sun Feb 12 00:13:27 CST 2017
go version go1.7.5 linux/amd64