令人激动!Go 泛型代码合入 master(附尝鲜方法)

大家好,我是慢一拍的后方记者煎鱼。

按照先前官方和文章的说法,Go 泛型预计是在 Go1.18 正式释出。

在 GopherCon 2020 Go Team AMA 时,要在今年底要能有生产环境的试用版上线,这是 rsc 所提出的一个管理目标。

转折点

近期出现了一个新的转折点,能够让大家在主干分支(master)上就能享受到泛型的功能。

而 master 分支对应了 Go1.17 的版本。因此未来将可以在 Go1.17 使用到泛型,这是一个比较惊喜的事情。

原因

这件事情为什么会突然发生呢?一切都得从背景说起。原本 Go 泛型是一直在 dev.typeparams 分支上进行研讨和开发。

由于泛型不是简单的一两个模块的代码变更,而是涉及大量的代码变更。

因此需要经常保持与 master 分支的代码同步(近两个月共 20+ 次),会涉及代码冲突/合并的处理,且对于一些冲突的模块他们也不熟悉,所以期望迁移到 master 分支上进行开发。

如何不影响既有功能

这类提前放入主版本的操作,在 Go 语言中并不少见。像是现在所见的 GO111MODULE,早期的 GO15VENDOREXPERIMENT 都有些这么个味道。都是逐步入场,分阶段使用,等确定成熟、完善后再渐渐去掉。

因此本次泛型也采取了这种方法,按照提案,目前使用的是 -G 标识做为泛型的开关。

计划如下:

  • -G=0:继续使用传统的类型检查器。
  • -G=1:使用 type2,但不支持泛型。
  • -G=2:使用 type2,支持泛型。

在完成 types2 的错误和现有的错误的开发协调后,计划在 Go 1.17 将 -G=1 设置为默认值。

未来也许可以在 Go 1.18 中放弃对 -G=0 的支持,这样后续在默认启用 -G=2 上会变得更容易。

在 Go1.17 尝鲜

在 Go1.17 尝鲜,也就意味着需要拉取 Go 语言的 master 分支的代码,Go1.17 现在正处于开发阶段:

我们可以通过 gotip 来达到下载 master 分支代码的目的:

$ go get golang.org/dl/gotip
$ gotip download
From https://go.googlesource.com/go
 * branch            master     -> FETCH_HEAD
   44361140c0..d9fd38e68b master     -> origin/master
Previous HEAD position was 44361140c0 embed: update docs for proposal tweaks
...

在拉取完毕后可以执行 gotip version 查看所拉取的版本(commit-id):

$ gotip version
go version devel +d9fd38e68b Sat Feb 27 03:03:29 2021 +0000 darwin/amd64

在确定 gotip 正常后,我们就可以编写泛型的示例代码了,如下:

func Print[T any](s []T) {
	for _, v := range s {
		fmt.Print(v)
	}
}

func main() {
	Print([]string{"脑子进, ", "煎鱼了\n"})
}

如果执行像往常那样执行,是会直接提示无法识别泛型的一些标识符:

$ gotip run main.go 
# command-line-arguments
./main.go:7:6: missing function body
./main.go:7:11: syntax error: unexpected [, expecting (

结合上文的解析,我们需要指定 -G 标识,就可以运行了。如下:

$ gotip run -gcflags=all=-G=3 main.go 
# command-line-arguments
./main.go:7:6: internal compiler error: Cannot export a generic function (yet): Print

显然,正确的走进泛型的逻辑里去了,虽然愉快的报错了,但 Matthew Dempsky 表示这很正常,毕竟 Go 泛型还在开发阶段。

可能会有的小伙伴注意到,-G 指定的是 3,与前文所述不符。这与早期的编码有关:

已经提了 CL 变更,只是代码冲突了,待解决。

总结

Go 语言的泛型开发计划已经比较明确。首先合入 master 分支,再逐步完成开发,逐步开放。

再进行 -G 默认值的调整,最后在泛型完善后就默认开启,把 -G 标识彻底去掉。

细品,与 Go modules 的方向是不是差不多。一开始 GO111MODULE 需要手动开启 on(也就是默认 off),再到 Go1.16 GO111MODULE 默认为 on

以此完成了一个正反馈的循环,逐步开放,接受社区反馈和开发调整。

结论,Go 泛型指日可待了



go

234 Words

2021-04-05 16:06 +0800