在前几天,Go1.16rc1 抢先发布了。结合常规的 28 发布规律,其将会在 2021.02 月份左右发布正式版本。
这次 Go1.16 也带来了一些新特性或变更。那么作为一个 Gopher,想必不能错过这次的更新。
今天这篇文章将会带大家了解一下 Go1.16 的几个需要关注的特性。
废弃 io/ioutil
Go 官方认为 io/ioutil 这个包的定义不明确且难以理解。所以 Russ Cox 在 2020.10.17 提出了废弃 io/ioutil 的提案。
大致变更如下:
Discard => io.Discard
NopCloser => io.NopCloser
ReadAll => io.ReadAll
ReadDir => os.ReadDir
ReadFile => os.ReadFile
TempDir => os.MkdirTemp
TempFile => os.CreateTemp
WriteFile => os.WriteFile
与此同时大家也不需要担心存在破坏性变更,因为有 Go1 兼容性的保证,在 Go1 中 io/ioutil 还会存在,只变更内部的方法调用:
func ReadAll(r io.Reader) ([]byte, error) {
return io.ReadAll(r)
}
func ReadFile(filename string) ([]byte, error) {
return os.ReadFile(filename)
}
大家在后续也可以改改调用习惯。
支持静态资源嵌入
如果我们希望把静态文件编译进 Go 的二进制文件的话,在以往需要借助 go-bindata/go-bindata 这类第三方开源库来实现。
而从 Go1.16 起,通过 go:embed
就可以快速实现这个功能:
import _ "embed"
//go:embed hello.txt
var s string
print(s)
通过对变量 s
声明 go:embed
指令,使其在编译时读取当前目录下的 hello.txt
文件。
最终变量 s
就会输出 hello.txt
文件中的字符串内容。
新增 io/fs 的支持
新增了标准库 io/fs,正式将文件系统相关的基础接口抽象到了该标准库中。
以前的话大多是在 os
标准库中,这一步抽离更进一步的抽象了文件树的接口。在后续的版本中,大家可以优先考虑使用 io/fs
标准库。
调整切片扩容策略
Go1.16 以前的 slice 的扩容条件是 len
,在最新的代码中,已经改为了以 cap
属性作为基准:
// src/runtime/slice.go
if cap > doublecap {
newcap = cap
} else {
// 这是以前的代码:if old.len < 1024 {
// 下面是 Go1.16rc1 的代码
if old.cap < 1024 {
newcap = doublecap
}
以官方的 test case 为例:
func main() {
const N = 1024
var a [N]int
x := cap(append(a[:N-1:N], 9, 9))
y := cap(append(a[:N:N], 9))
println(cap(x), cap(y))
}
在 Go1.16 以前输出 2048, 1280。在 Go1.16 及以后输出 1280, 1280,保证了两种的一致。
支持 Apple Silicon M1
众所周知,最新版本的 Mac 采用了新的 64 位 ARM 架构,因此在 Go1.16 后正式支持了 GOOS=darwin
和 GOARCH=arm64
。
而相应的先前用于 iOS 端口的,将改为 GOOS=ios
和 GOARCH=arm64
。
同时 Apple M1 能不能很好的跑好 Go 语言程序也是各大微信群爱讨论的问题,在 GoLand 上:
需要注意,GoLand 的一些与 Go1.16 相关的特性要到后续新版本才可以使用。
调整 Go modules 策略
从 Go1.16 起,Go modules 的环境变量 GO111MODULE
默认开关将为 on
,不再是之前是 auto
了。
还在使用 GOPATH,或 Go modules 没切全的同学这一块需要特别注意。
新增 GODEBUG inittrace
GODEBUG 新增 inittrace 指令,可以用于 init
方法的排查:
$ GODEBUG=inittrace=1 go run main.go
输出结果:
init internal/bytealg @0.008 ms, 0 ms clock, 0 bytes, 0 allocs
init runtime @0.059 ms, 0.026 ms clock, 0 bytes, 0 allocs
init math @0.19 ms, 0.001 ms clock, 0 bytes, 0 allocs
init errors @0.22 ms, 0.004 ms clock, 0 bytes, 0 allocs
init strconv @0.24 ms, 0.002 ms clock, 32 bytes, 2 allocs
init sync @0.28 ms, 0.003 ms clock, 16 bytes, 1 allocs
init unicode @0.44 ms, 0.11 ms clock, 23328 bytes, 24 allocs
...
主要作用是 init 函数跟踪的支持,以用于 init 调试和启动时间的概要分析,算是一个 GODEBUG 的补充功能点。
简化结构体标签
在 Go 语言的结构体中,我们常常会因为各种库的诉求,需要对结构体的 tag
设置标识。
如果像是以前,量比较多就会变成:
type MyStruct struct {
Field1 string `json:"field_1,omitempty" bson:"field_1,omitempty" xml:"field_1,omitempty" form:"field_1,omitempty" other:"value"`
}
但在 Go1.16 及以后,就可以通过合并的方式:
type MyStruct struct {
Field1 string `json,bson,xml,form:"field_1,omitempty" other:"value"`
}
方便和简洁了不少。
总结
在本次 Go1.16 中带来了不少小优化和新的特性支持。离 Go1.18 的泛型又近了一步。
另外在本次新版本中,像是 template
支持跨行:
{{"hello" |
printf}}
又或是 Linux 的默认内存管理策略下又从 MADV_FREE 改回了 MADV_DONTNEED 策略,大家在新版本中不再需要设置:
GODEBUG=madvdontneed=1
大家若有需求都可以进一步去了解,现在新版本的功能特性已经锁定,基本尘埃落定。
传送门:https://tip.golang.org/doc/go1.16。
分享 Go 语言、微服务和奇怪的系统设计
长按关注煎鱼,在知识的海洋里遨游: