Go 1.18新特性 支持泛型与性能优化

距离 Go 1.17 发布七个月后,Go 1.18 正式发布了!Go 1.18 是一个包含大量新功能的版本,包括对语言本身做了有史以来最大的改变(泛型)、工具链的实现、运行时和库的更改,还改善了性能。

与往常一样,该版本保持了 Go 1的兼容性承诺: 几乎所有 Go 程序都能像以前一样继续编译和运行。下面来看一下新版本的一些重大特性:

泛型

以下是关于 Go 1.18 泛型的最明显变化的列表:

函数和类型声明的语法,现在接受 类型参数 。
参数化函数和类型可以通过 在方括号中 列出类型参数来实例化。
新标记 ~ 已添加到 操作符和标点符号 中。
接口类型的语法现在允许嵌入任意类型(不仅仅是接口的类型名称)以及 union 和 ~T 类型元素。,这样的接口只能用作类型约束。
新的 预声明标识符 any 是空接口的别名,可以用来代替 interface{} .
新的 预声明标识符 comparable 是一个接口,表示可以使用 == 或者 != 比较的所有类型的集合,它只能用作(或嵌入)类型约束。
有三个使用泛型的实验包可能有用,这些包在 x/exp 存储库中;但它们的 API 不在 Go 1 兼容性承诺的保证范围内:

golang.org/x/exp/constraints :对通用代码有用的约束,例如 constraints.Ordered .
golang.org/x/exp/slices :对任何元素类型的切片进行操作的通用函数集合。
golang.org/x/exp/maps :对任何键或元素类型的映射进行操作的通用函数集合。
当前的泛型实现具有以下已知限制:

Go 编译器无法处理泛型函数或方法中的类型声明,计划在 Go 1.19 中取消这个限制。
Go 编译器不接受具有预声明函数 real、imag 和 complex 的参数类型的参数,计划在 Go 1.19 中取消这个限制。
如果 m 由 P 的约束接口显式声明,Go 编译器仅支持在类型参数类型 P 的值 x 上调用方法 m。类似地,方法值 x.m 和方法表达式 P.m 也仅在 m 由 P 显式声明时才受支持,即使 m 可能在 P 的方法集中,因为 P 中的所有类型都实现了 m,计划在 Go 1.19 中取消这个限制。
Go 编译器不支持访问结构字段 x.f,其中 x 是类型参数类型,即使类型参数的类型集中的所有类型都具有字段 f,计划在 Go 1.19 中取消这个限制。
不允许将类型参数或指向类型参数的指针作为结构类型中的未命名字段嵌入,同样地,也不允许在接口类型中嵌入类型参数。
具有多个 term 的 union 元素可能不包含具有非空方法集的接口类型。
泛型代表 Go 生态系统的巨大变化,虽然官方更新了几个支持泛型的核心工具,但还有很多工作要做。剩余的工具、文档和库需要一些时间才能赶上这些语言变化。此外,官方公告中还有这么一段话:

可能会有一些使用泛型的代码可以在 1.18 版本中使用,但在以后的版本中会中断。
我们不计划或期望做出任何此类更改,但是,由于我们今天无法预见的原因,可能需要 在未来版本中破坏 1.18 的程序 。
我们鼓励在有意义的地方使用泛型,但在生产环境中部署泛型代码时,请谨慎行事。
(虽然泛型是搞出来了,但很可能有 Bug,不建议在生产中使用)

模糊测试

Go 1.18 包括 fuzzing(模糊测试) 的实现,如 fuzzing 提案 所述,详情请参阅 fuzzing 教程 以开始使用。
注意,模糊测试会消耗大量内存 ,并且可能会影响机器运行时的性能。
另请注意,模糊引擎在运行时会将扩展测试覆盖率的值写入模糊缓存目录 $GOCACHE/fuzz 。目前对可以写入模糊缓存的文件数量或总字节数 没有限制 ,因此可能会占用大量存储空间(可能为 GB 级别)。
编译器
现在编译器可以内联包含范围循环或标记为循环的函数。
编译器的类型检查器被完全替换以支持泛型,一些错误消息可能使用与以前不同的措辞(提供更多详细信息,或以更有用的方式表述)。
由于与支持泛型相关的编译器的更改,Go 1.18 的 编译速度 可能比 Go 1.17 的编译速度 慢大约 15% ,代码的执行时间不受影响,目前计划在 Go 1.19 中提高编译器的速度。
Bug fixes
Go 1.18 编译器可以正确地报告在函数文本中设置但从未使用过的变量的错误(已声明但未使用),解决了一个老问题 issue #8560 。
Go 1.18 编译器现在在将如 ‘1’ << 32 之类的符文常量表达式作为参数传递给预声明函数 print 和 println 时会报告溢出。
Ports AMD64
Go 1.18 引入了新的GOAMD64环境变量,它在编译时选择 AMD64 架构的最低目标版本,允许的值为v1、 v2、v3或v4,默认是v1。

RISC-V

Linux 上的 64 位 RISC-V 架构(linux/riscv64 端口)现在支持 c-archive 和 c-shared 构建模式。

Linux

Go 1.18 需要 Linux 内核版本 2.6.32 或更高版本。

Windows

windows/arm 和 windows/arm64 端在支持非合作抢占,有希望解决在调用 Win32 函数时遇到的一些细微的 bug,这些bug在很长一段时间内会阻塞。

iOS

在 iOS(ios/arm64 端口)和在基于 AMD64 的 macOS(ios/amd64 端口)上运行的 iOS 模拟器上,Go 1.18 现在需要 iOS 12 或更高版本;已停止支持以前的版本。

FreeBSD

Go 1.18 是支持 FreeBSD 11.x 的最后一个版本,Go 1.19 需要 FreeBSD 12.2+ 或 FreeBSD 13.0+。

性能提升

由于 Go1.17 中寄存器 ABI 调用约定扩展到了 RM64 / Apple M1 / PowerPC 64 架构,因此 Go1.18 对这几个架构包含了高达 20% 的 CPU 性能提升。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值