Go1.17 新特性:新版构建约束

大家好,我是 polarisxu。

Go 1.17 下个月就要正式发布了。很多人要问泛型了吧,泛型已经很明确了,Go1.18 会有。今天给大家介绍 Go1.17 的一个新特性:构建约束 — Build Constraints。

确切来说,这个特性相关的工作在 1.16 时就加入,但处于过度阶段,1.17 在各方面都更完善,更完整的支持,是时候了解它了。

01 什么是构建约束

构建约束(build constraint),也叫做构建标记(build tag),是在 Go 源文件最开始的注释行,比如:

// +build linux

看到这个,相信很多人都不陌生,因为这是 Go 一开始就有的特性,在 Go 源码中有很多这样的注释行。上面注释行的意思,这个文件只在 Linux 系统会包含在包中,其他系统会忽略这个文件。

几个注意点:

  • 约束可以出现在任何源文件中,比如 .go.s 等;
  • 必须在文件顶部附近,它的前面只能有空行或其他注释行;可见包子句也在约束之后;
  • 约束可以有多行;
  • 为了区别约束和包文档,在约束之后必须有空行;

针对某个构建约束,可使用的词如下:

  • 特定操作系统,对应 runtime.GOOS 的可用值,比如 linux、windows 等;
  • 特定的架构,对应 runtime.GOARCH 的可用值,比如 386、amd64 等;
  • 使用的编译器,比如 gc、gccgo;
  • 支持 cgo 命令时,可以使用 cgo;
  • Go 的主要发布版本,比如 go1.17、go1.16 等;(测试版本和 fixbug 版本不支持)
  • 自定义的 tag,编译时通过 -tags 传递的值;
  • 可以加入任意值,一般用 ignore 来忽略构建;

此外,文件名可以通过 GOOS 和 GOARCH 来做构建约束。

02 旧版构建约束

从上面看到,构建约束的语法是 // +build 这种形式,如果多个条件组合,通过空格、逗号或多行构建约束表示。比如:

// +build linux,386

你知道什么意思吗?表示在 linux AND 386。逗号表示 AND,空格表示 OR。那看一个复杂的:

// +build linux,386 darwin,!cgo

是不是有点懵?我也有点懵!它表示的意思是:(linux AND 386) OR (darwin AND (NOT cgo)) 。

有些时候,多个约束分成多行书写,会更易读些:

// +build linux darwin
// +build amd64

这相当于:(linux OR darwin) AND amd64 。

是不是很复杂,很难记忆?

正因为太复杂,很容易出错。而且,Go 中有不少注释是有特殊意义的,也为了一致性考虑,因此有了新版的构建约束。

03 新版构建约束

在 Go 源码中,经常会见到类似下面开头的注释:

//go:link

新版的构建约束,也使用了 //go: 开头:

//go:build

注意 // 和 go 之间不能有空格。

同时新版语法使用布尔表达式,而不是逗号、空格等。布尔表达式,会更清晰易懂,出错可能性大大降低。

比如旧语法:

// +build linux,386

对应的新语法:

//go:build linux && 386

构建标记的基础语法与其当前形式没有变化,但是构建标记的组合现在是用 Go 的 || 、 && 和 ! 运算符和括号。(请注意,构建标记并不总是有效的 Go 表达式,即使它们共享操作符,因为标记并不总是有效的标识符。例如:”go1.1"。)

新语法可以使用 Go spec 的 EBNF 标记来表示:

BuildLine      = "//go:build" Expr
Expr           = OrExpr
OrExpr         = AndExpr   { "||" AndExpr }
AndExpr        = UnaryExpr { "&&" UnaryExpr }
UnaryExpr      = "!" UnaryExpr | "(" Expr ")" | tag
tag            = tag_letter { tag_letter }
tag_letter     = unicode_letter | unicode_digit | "_" | "."

采用新语法后,一个文件只能有一行构建语句,而不是像旧版那样有多行。这样可以避免多行的关系到底是什么的问题。

Go1.17 中,gofmt 工具会自动根据旧版语法生成对应的新版语法,为了兼容性,两者都会保留。比如原来是这样的:

// +build !windows,!plan9

执行 Go1.17 的 gofmt 后,变成了这样:

//go:build !windows && !plan9
// +build !windows,!plan9

如果文件中已经有了这两种约束形式,gofmt 会根据 //go:buid 自动覆盖 // +build 的形式,确保两者表示的意思一致。如果只有新版语法,不会自动生成旧版的,这时,你需要注意,它不兼容旧版本了。

另外,Vet 工具现在能够检测出两种语法的不一致。所以,建议大家在编辑器中保存文件时自动执行 gofmt。

早在 Go1.16 时就新增了一个包:go/build/constraint ,专门处理新版构建约束。

关于新版约束的设计文档请移步:https://go.googlesource.com/proposal/+/master/design/draft-gobuild.md

04 总结

新版本的构建约束可读性更强,更容易书写,不容易出错。有兴趣的可以自己针对构建约束,同时书写两种形式,体会下新版的好处。

最后提醒一点,新版约束中,一定要注意 // 和 go 之间不能有空格!