Go语言演进的双保险:GOEXPERIMENT与GODEBUG

39e3bd7045ec1cb3446a2e1d9fcf79d8.jpeg请点击上方蓝字TonyBai订阅公众号!

2e38c5e83f7aac4fcaca092e1e3c9b71.png

(建议用浏览器扫描上面二维码)

Go语言自诞生以来就以其简洁、高效和强大的并发支持而闻名,Go团队承诺保持Go1向后兼容性[1],以确保用户的代码在未来的版本中继续正常运行。然而,保持语言的稳定性与不断创新(增加新特性)之间的平衡一直是Go团队面临的挑战。为了应对这一挑战,Go语言引入了两个关键机制:GOEXPERIMENT和GODEBUG来平衡新功能的试验、稳定发布和向后兼容。这两个机制共同构成了Go语言特性发布的“双保险”,确保语言能够稳步前进的同时,不会因为激进的改变而影响现有代码的稳定性。本文就来简单探讨一下这两个机制是如何保障Go语言新特性稳定发布的。

1. GOEXPERIMENT:新特性的摇篮

GOEXPERIMENT是一个Go语言的环境变量,是用于控制实验性特性的机制。它允许开发者在编译时(使用go build、go install、go run或go test)启用一些尚未正式发布的语言特性或优化。通过GOEXPERIMENT,Go团队能够在正式发布之前广泛测试新功能,收集反馈并进行必要的调整。

比如,在今年8月发布的Go 1.23版本[2]发布了一个实验特性:带有类型参数的type alias[3],就像下面代码一样,我们可以在编译时开启该实验特性:

// github.com/bigwhite/experiments/blob/master/go1.23-examples/lang/generic_type_alias.go

$GOEXPERIMENT=aliastypeparams go build generic_type_alias.go
$./generic_type_alias
Int Slice: [1 2 3 4 5]
String Slice: [hello world]
Person Slice: [{Alice 30} {Bob 25}]

如果不开启实验特性,上述的代码就会编译失败:

// github.com/bigwhite/experiments/blob/master/go1.23-examples/lang/generic_type_alias.go

$go build generic_type_alias.go
# command-line-arguments
./generic_type_alias.go:5:6: generic type alias requires GOEXPERIMENT=aliastypeparams

我们看到:通过设置GOEXPERIMENT=featureflag可以开启对应的实验特性,如果要同时开启多个实验特性,可以用逗号分隔的实验特性列表,就像下面这样:

$GOEXPERIMENT=featureflag1,featureflag2,...,featureflagN go build

那么如何查看当前Go版本有哪些实验验特性可用呢?我们可以借助go doc[4]工具,以go 1.23.0为例:

$go doc goexperiment.Flags
package goexperiment // import "internal/goexperiment"

type Flags struct {
 FieldTrack        bool
 PreemptibleLoops  bool
 StaticLockRanking bool
 BoringCrypto      bool

 // RegabiWrappers enables ABI wrappers for calling between
 // ABI0 and ABIInternal functions. Without this, the ABIs are
 // assumed to be identical so cross-ABI calls are direct.
 RegabiWrappers bool
 // RegabiArgs enables register arguments/results in all
 // compiled Go functions.
 //
 // Requires wrappers (to do ABI translation), and reflect (so
 // reflection calls use registers).
 RegabiArgs bool

 // HeapMinimum512KiB reduces the minimum heap size to 512 KiB.
 //
 // This was originally reduced as part of PacerRedesign, but
 // has been broken out to its own experiment that is disabled
 // by default.
 HeapMinimum512KiB bool

 // CoverageRedesign enables the new compiler-based code coverage
 // tooling.
 CoverageRedesign bool

 // Arenas causes the "arena" standard library package to be visible
 // to the outside world.
 Arenas bool

 // CgoCheck2 enables an expensive cgo rule checker.
 // When this experiment is enabled, cgo rule checks occur regardless
 // of the GODEBUG=cgocheck setting provided at runtime.
 CgoCheck2 bool

 // LoopVar changes loop semantics so that each iteration gets its own
 // copy of the iteration variable.
 LoopVar bool

 // CacheProg adds support to cmd/go to use a child process to implement
 // the build cache; see https://github.com/golang/go/issues/59719.
 CacheProg bool

 // NewInliner enables a new+improved version of the function
 // inlining phase within the Go compiler.
 NewInliner bool

 // RangeFunc enables range over func.
 RangeFunc bool

 // AliasTypeParams enables type parameters for alias types.
 // Requires that gotypesalias=1 is set with GODEBUG.
 // This flag will be removed with Go 1.24.
 AliasTypeParams bool
}
    Flags is the set of experiments that can be enabled or disabled in the
    current toolchain.

    When specified in the GOEXPERIMENT environment variable or as build tags,
    experiments use the strings.ToLower of their field name.

    For the baseline experimental configuration, see objabi.experimentBaseline.

    If you change this struct definition, run "go generate".

go doc输出结果中的Flags结构体其实是 $GOROOT/internal/goexperiment包中的一个类型,这个类型每一个字段对应一个实验特性,字段名的小写即可作为GOEXPERIMENT的值,比如AliasTypeParams的小写形式aliastypeparams正是我们在前面示例中使用的实验特性。

在Flags结构体中,我们看到了几个十分熟悉的字段,比如LoopVar、RangeFunc、Arenas等,这些实验特性有些已经正式落地,比如:Go 1.21[5]引入的实验特性Loopvar[6]在Go 1.22版本[7]中成为正式语法特性。而Arenas[8]这个在Go 1.20版本[9]引入的实验特性则因为实现上缺陷而迟迟不能转正,目前处于proposal hold状态[10]

Go对实验特性的引入分为两种情况:

  • 默认开启实验特性,无需在编译时通过GOEXPERIMENT=featureflag显式开启

在Go 1.22中的exectracer2就是这样一个实验特性,它控制着是否使用新的execution trace[11]的实现。

对于这样的实验特性,我们可以通过GOEXPERIMENT=nofeatureflag对其进行显式关闭,以Go 1.22引入的实验特性ExecTracer2为例,可以使用下面命令关闭该实验特性:

$GOEXPERIMENT=noexectracer2 go build

注:之后使用go version your-go-app,可以看到“your-go-app: go1.22.0 X:noexectracer2”的输出。

  • 默认不开启实验特性,需在编译时通过GOEXPERIMENT=featureflag显式开启

这就是我们最熟悉的实验特性引入方式,Go 1.23的AliasTypeParams实验特性就是默认不开启的,前面的例子已经给出了开发方法,这里就不赘述了。

实验特性通常经过1到2个版本的实验便会落地,成为正式特性。已经落地的实验特性通常会从Flags结构体中移除,比如Go 1.22的goexperiment.Flags结构体中的ExecTracer2,在Go 1.23中就看不到了。但总有一些已经落地的实验特性对应的flag字段依然还留存在Flags结构体里,比如:LoopVar,这个原因还不得而知!并且这样的已经成为正式特性的Flag,我们也无法再通过GOEXPERIMENT=nofeatureflag对其进行显式关闭了,因为它已经不再是实验特性了!

不过有些实验特性即便转正落地了,也会考虑到新特性对legacy code行为的影响而去读取go.mod中的go version再决定是否应用新特性,比如LoopVar。LoopVar转正后,该特性也仅在编译的包来自于包含声明Go 1.22或更高版本的模块时适用,比如:Go 1.22或Go 1.23。这可以确保没有程序会因为简单地采用新的Go版本而改变行为,我们来看一个例子:

// go.mod

module demo

go 1.20

// main.go
package main

import (
 "fmt"
 "time"
)

func main() {
 var m = [...]int{1, 2, 3, 4, 5}

 for i, v := range m {
  go func() {
   time.Sleep(time.Second * 3)
   fmt.Println(i, v)
  }()
 }

 time.Sleep(time.Second * 5)
}

我们使用go 1.23.0版本编译该包,并运行输出的程序:

$go build 
$./demo   
4 5
4 5
4 5
4 5
4 5

可以看到,即便使用了Go 1.23版本,但因当前module的go version依然是go 1.20,Go编译器默认不会开启loopvar特性。

不过如果我们显式使用GOEXPERIMENT=loopvar,go编译器便不会考虑go.mod文件中的go version是什么版本,都会开启loopvar新特性:

$GOEXPERIMENT=loopvar go build 
$./demo                        
4 5
1 2
0 1
2 3
3 4

Go编译器会有一套Go试验特性的默认值,如果你通过GOEXPERIMENT显式开启了某些特性,导致该特性flag值与默认值不同,那么我们可以通过go version命令查看到这些不同之处。以上面GOEXPERIMENT=loopvar go build构建出的demo为例:

$go version demo               
demo: go1.23.0 X:loopvar

目前Go官方尚没有一个专门的页面用于汇总GOEXPERIMENT的各个flag的随Go版本release的历史,我们只能通过Flag字段在go issues查找其对应的issue来重温当时的情况。

到这里,我们可以看到GOEXPERIMENT引入的实验特性机制可以让Go团队相对稳健的向Go语言引入新特性(虽然不是所有新特性都需要走式样特性的流程,比如对泛型的支持[12]等),但是当新特性破坏了向后兼容,或者Go团队要对现有特性的错误语义(比如panicnil[13])进行变更时,Go1这个严格的兼容性规则就很可能成为阻碍在大家面前的一道门槛!为了在保持兼容性和推动创新之间取得平衡,Go团队就需要一种新的机制,通过渐进式的方法来引入破坏性(break change)的变更,这就是GODEBUG控制机制,下面我们就来说说GODEBUG。

2. GODEBUG:在运行时控制特性行为的开关

GODEBUG也是一个Go环境变量,和GOEXPERIMENT用于构建时不同,GODEBUG用在运行时控制Go程序的某些行为。它允许开发者临时将某一特性恢复到旧的行为,即使在新版本中该特性的默认行为已经发生了改变。

GODEBUG的设置形式为逗号分隔的key=value对,例如:

$GODEBUG=http2client=0,http2server=0 ./your-go-app

这个设置会禁用客户端和服务器端对HTTP/2的使用。

上面是使用GODEBUG禁用新特性的例子。对于存量特性语义或实现变更,比如Go 1.23版本对time.Timer和Ticker进行了重实现,新实现底层使用了无缓冲channel,但通过下面设置可以恢复原先实现中的带缓冲channel:

$GODEBUG=asynctimerchan=1 ./your-go-app

考虑到兼容性而进行的GODEBUG设置将在至少两年(四个Go版本)内保持。但一些设置,例如http2client和http2server,将会更长时间地保持,甚至是无限期的。

除了GODEBUG环境变量之外,Go还提供了其他几种进行特性行为设置的方式,下面我们来看看。

3. GODEBUG、go:debug和go.mod中godebug directive的关系

3.1. //go:debug指令

从Go 1.21开始,可以在源代码中使用//go:debug指令来设置GODEBUG的值。这些指令必须放在文件的顶部,在package语句之前。例如:

//go:debug panicnil=1
//go:debug asynctimerchan=0
package main

这些指令会在编译时被处理,并影响生成的二进制文件的行为。

3.2 go.mod中的godebug指令

从Go 1.23开始,可以在go.mod文件中使用godebug指令来设置GODEBUG的默认值,例如:

// go.mod

godebug (
    default=go1.21
    panicnil=1
    asynctimerchan=0
)

这个配置会影响整个模块(module)的默认GODEBUG设置。

3.3 优先级和应用范围

那么GODEBUG、//go:debug以及go.mod中的godebug指令的优先级关系是怎样的呢?

显然,环境变量GODEBUG优先级最高,因为它可以在运行时覆盖其他设置,适用于临时调试或特定运行环境。

go:debug指令优先级次之,通常应用于特定的main包,适用于对特定程序进行精细控制。

而go.mod中的godebug指令优先级最低,为整个模块设置默认值,适用于项目级别的配置。

基于上述关系,我们来看看一个Go应用GODEBUG设置的默认值的确定过程。当没有显示设置GODEBUG环境变量时,各设置的默认值按以下顺序确定:

  • 首先查看用于构建程序的Go工具链(版本)的默认值。

  • 然后根据go.mod或go.work中声明的Go版本(go version)进行调整。

  • 之后应用go.mod中的godebug指令(如果有的话)。

  • 最后是//go:debug,通常仅应用于main module。

例如,如果一个项目的go.mod声明了go 1.20,那么即使使用Go 1.21工具链编译,也会默认使用panicnil=1(即允许panic(nil))。

不过有特殊情况需要注意,比如对于声明早于Go 1.20版本的项目,GODEBUG默认值会被配置为匹配Go 1.20的行为,而不是更早的版本;又比如在测试环境中,*_test.go文件中的//go:debug指令会被视为测试主包的指令等。

这么看规则还是蛮复杂的,那么编译后待执行的程序的默认GODEBUG的设置究竟是什么呢?我们可以通过go version -m来查看,以gopls v0.16.2为例:

$go version -m /Users/tonybai/Go/bin/gopls
/Users/tonybai/Go/bin/gopls: go1.23.0
 path golang.org/x/tools/gopls
 mod golang.org/x/tools/gopls v0.16.2 h1:K1z03MlikHfaMTtG01cUeL5FAOTJnITuNe0TWOcg8tM=
 dep github.com/BurntSushi/toml v1.2.1 h1:9F2/+DoOYIOksmaJFPw1tGFy1eDnIJXg+UHjuD8lTak=
 dep github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
 dep golang.org/x/exp/typeparams v0.0.0-20221212164502-fae10dda9338 h1:2O2DON6y3XMJiQRAS1UWU+54aec2uopH3x7MAiqGW6Y=
 dep golang.org/x/mod v0.20.0 h1:utOm6MM3R3dnawAiJgn0y+xvuYRsm1RKM/4giyfDgV0=
 dep golang.org/x/sync v0.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ=
 dep golang.org/x/telemetry v0.0.0-20240829154258-f29ab539cc98 h1:Wm3cG5X6sZ0RSVRc/H1/sciC4AT6HAKgLCSH2lbpR/c=
 dep golang.org/x/text v0.16.0 h1:a94ExnEXNtEwYLGJSIUxnWoxoRz/ZcCsV63ROupILh4=
 dep golang.org/x/tools v0.22.1-0.20240829175637-39126e24d653 h1:6bJEg2w2kUHWlfdJaESYsmNfI1LKAZQi6zCa7LUn7eI=
 dep golang.org/x/vuln v1.0.4 h1:SP0mPeg2PmGCu03V+61EcQiOjmpri2XijexKdzv8Z1I=
 dep honnef.co/go/tools v0.4.7 h1:9MDAWxMoSnB6QoSqiVr7P5mtkT9pOc1kSxchzPCnqJs=
 dep mvdan.cc/gofumpt v0.6.0 h1:G3QvahNDmpD+Aek/bNOLrFR2XC6ZAdo62dZu65gmwGo=
 dep mvdan.cc/xurls/v2 v2.5.0 h1:lyBNOm8Wo71UknhUs4QTFUNNMyxy2JEIaKKo0RWOh+8=
 build -buildmode=exe
 build -compiler=gc
 build DefaultGODEBUG=asynctimerchan=1,gotypesalias=0,httplaxcontentlength=1,httpmuxgo121=1,httpservecontentkeepheaders=1,panicnil=1,tls10server=1,tls3des=1,tlskyber=0,tlsrsakex=1,tlsunsafeekm=1,winreadlinkvolume=0,winsymlink=0,x509keypairleaf=0,x509negativeserial=1
 build CGO_ENABLED=1
 build CGO_CFLAGS=
 build CGO_CPPFLAGS=
 build CGO_CXXFLAGS=
 build CGO_LDFLAGS=
 build GOARCH=amd64
 build GOOS=darwin
 build GOAMD64=v1

我们看到其DefaultGODEBUG如下:

DefaultGODEBUG=asynctimerchan=1,gotypesalias=0,httplaxcontentlength=1,httpmuxgo121=1,httpservecontentkeepheaders=1,panicnil=1,tls10server=1,tls3des=1,tlskyber=0,tlsrsakex=1,tlsunsafeekm=1,winreadlinkvolume=0,winsymlink=0,x509keypairleaf=0,x509negativeserial=1

相对于GOEXPERIMENT的flags的数量,GODEBUG的设置项更多,下面我们根据go官方资料整理一个GODEBUG设置项列表供大家参考(信息截至2024.10.7)。

4. GODEBUG设置的历史演进

下表按照Go版本顺序列出了各个GODEBUG设置,包括它们被引入的版本、含义以及如何开启和关闭它们:

Go版本设置名称含义开启方法关闭方法
1.5netdns控制纯Go DNS解析器的使用GODEBUG=netdns=goGODEBUG=netdns=cgo
1.6http2client控制HTTP客户端的HTTP/2支持GODEBUG=http2client=1 (默认)GODEBUG=http2client=0
1.6http2server控制HTTP服务器的HTTP/2支持GODEBUG=http2server=1 (默认)GODEBUG=http2server=0
1.6http2debug启用HTTP/2调试日志GODEBUG=http2debug=1GODEBUG=http2debug=0 (默认)
1.10gocacheverify验证构建缓存的结果GODEBUG=gocacheverify=1GODEBUG=gocacheverify=0 (默认)
1.10gocachehash使用额外的哈希函数来计算构建缓存键GODEBUG=gocachehash=1GODEBUG=gocachehash=0 (默认)
1.10gocachetest缓存测试结果GODEBUG=gocachetest=1GODEBUG=gocachetest=0 (默认)
1.18x509sha1在X.509证书中允许使用SHA1GODEBUG=x509sha1=1GODEBUG=x509sha1=0 (默认)
1.19execerrdot使路径查找将当前目录视为可执行文件的错误GODEBUG=execerrdot=1 (默认)GODEBUG=execerrdot=0
1.19netedns0禁止在DNS请求中发送EDNS0附加标头GODEBUG=netedns0=0GODEBUG=netedns0=1 (默认)
1.20tarinsecurepath允许在tar存档中使用不安全路径GODEBUG=tarinsecurepath=1 (默认)GODEBUG=tarinsecurepath=0
1.20zipinsecurepath允许在zip存档中使用不安全路径GODEBUG=zipinsecurepath=1 (默认)GODEBUG=zipinsecurepath=0
1.20randautoseed控制math/rand全局随机数生成器的自动播种GODEBUG=randautoseed=1 (默认)GODEBUG=randautoseed=0
1.20x509usefallbackroots在证书验证期间使用备用根证书GODEBUG=x509usefallbackroots=1GODEBUG=x509usefallbackroots=0 (默认)
1.20installgoroot恢复预安装的.a文件的安装和使用GODEBUG=installgoroot=1GODEBUG=installgoroot=0 (默认)
1.21panicnil允许panic(nil)GODEBUG=panicnil=1GODEBUG=panicnil=0 (默认)
1.21jstmpllitinterp在ECMAScript 6模板字面量中禁用html/template操作GODEBUG=jstmpllitinterp=1 (默认)GODEBUG=jstmpllitinterp=0
1.21multipartmaxheaders设置MIME头的最大数量限制GODEBUG=multipartmaxheaders=<数值>默认无限制
1.21multipartmaxparts设置多部分表单的最大部分数限制GODEBUG=multipartmaxparts=<数值>默认无限制
1.21multipathtcp控制多路径TCP的使用GODEBUG=multipathtcp=1GODEBUG=multipathtcp=0 (默认)
1.22tlsmaxrsasize控制TLS握手中可接受的最大RSA密钥大小GODEBUG=tlsmaxrsasize=<比特数>默认为8192位
1.22httplaxcontentlength允许HTTP请求或响应中的空Content-Length标头GODEBUG=httplaxcontentlength=1GODEBUG=httplaxcontentlength=0 (默认)
1.22httpmuxgo121控制ServeMux的行为变化GODEBUG=httpmuxgo121=1GODEBUG=httpmuxgo121=0 (默认)
1.22gotypesalias控制类型检查器是否生成Alias类型GODEBUG=gotypesalias=1GODEBUG=gotypesalias=0 (默认)
1.22tls10server将服务器支持的最低TLS版本恢复到1.0GODEBUG=tls10server=1GODEBUG=tls10server=0 (默认)
1.22tlsrsakex恢复使用基于RSA密钥交换的默认TLS密码套件GODEBUG=tlsrsakex=1GODEBUG=tlsrsakex=0 (默认)
1.22tlsunsafeekm在不支持TLS 1.3或扩展主密钥的连接中重新启用ConnectionState.ExportKeyingMaterialGODEBUG=tlsunsafeekm=1GODEBUG=tlsunsafeekm=0 (默认)
1.22disablethp为Go内存禁用透明大页面GODEBUG=disablethp=1GODEBUG=disablethp=0 (默认)
1.22runtimecontentionstacks启用运行时锁的完整堆栈跟踪GODEBUG=runtimecontentionstacks=1GODEBUG=runtimecontentionstacks=0 (默认)
1.22x509usepolicies使用Certificate.Policies字段进行OID编组GODEBUG=x509usepolicies=1GODEBUG=x509usepolicies=0 (默认)
1.23asynctimerchan控制time包创建的通道是否为异步(缓冲)GODEBUG=asynctimerchan=1GODEBUG=asynctimerchan=0 (默认)
1.23winsymlink控制Windows上重解析点的模式位报告GODEBUG=winsymlink=0GODEBUG=winsymlink=1 (默认)
1.23winreadlinkvolume控制os.Readlink和filepath.EvalSymlinks是否尝试将卷规范化为驱动器号GODEBUG=winreadlinkvolume=0GODEBUG=winreadlinkvolume=1 (默认)
1.23tlskyber控制是否默认启用X25519Kyber768Draft00后量子密钥交换机制GODEBUG=tlskyber=0GODEBUG=tlskyber=1 (默认)
1.23x509negativeserial允许x509.ParseCertificate接受负序列号GODEBUG=x509negativeserial=1GODEBUG=x509negativeserial=0 (默认)
1.23tls3des恢复使用包含3DES密码套件的默认TLS密码套件GODEBUG=tls3des=1GODEBUG=tls3des=0 (默认)
1.23x509keypairleaf控制tls.X509KeyPair和tls.LoadX509KeyPair是否填充返回的tls.Certificate的Leaf字段GODEBUG=x509keypairleaf=0GODEBUG=x509keypairleaf=1 (默认)
1.23httpservecontentkeepheaders控制ServeContent在提供错误时是否保留某些响应头GODEBUG=httpservecontentkeepheaders=1GODEBUG=httpservecontentkeepheaders=0 (默认)

不过请注意以下几点:

  • 默认值可能会随着Go版本的更新而改变。

  • 某些设置可能在未来的Go版本中被移除。

  • 部分设置(如tlsmaxrsasize)允许指定具体的数值,而不仅仅是0或1。

  • 有些设置(如multipartmaxheaders和multipartmaxparts)在默认情况下是无限制的,需要明确设置一个数值来启用限制。

5. 小结

在Go语言的演进过程中,GOEXPERIMENT和GODEBUG两个机制起到了至关重要的作用。GOEXPERIMENT为新特性的实验和测试提供了灵活的环境,使得开发者可以在正式发布之前尝试和反馈新功能,从而确保Go语言的创新不会影响到已有代码的稳定性。通过这种方式,Go团队能够逐步引入新特性,同时维持向后兼容性。

另一方面,GODEBUG则为开发者提供了在运行时控制特性行为的工具,使得新版本引入的破坏性更改能够被临时禁用。这种灵活性使得开发者有一个平滑过渡的机会,能够在更新的同时,保证应用的平稳运行,避免了因语言更新而导致的潜在问题,使Go能够在保持稳定性的同时不断创新。

总的来说,这两个机制共同构成了Go语言特性发布的“双保险”,确保了语言的持续发展与稳定性之间的平衡。这一策略不仅促进了Go语言的创新,也增强了开发者的信心,使其能够在不断变化的环境中有效地编写和维护代码。

6. 参考资料

  • Go, Backwards Compatibility, and GODEBUG[14] - https://go.dev/doc/godebug

  • Proposal: Extended backwards compatibility for Go[15] - https://go.googlesource.com/proposal/+/master/design/56986-godebug.md


参考资料

[1] 

Go1向后兼容性: https://go.dev/doc/go1compat

[2] 

Go 1.23版本: https://tonybai.com/2024/08/19/some-changes-in-go-1-23

[3] 

带有类型参数的type alias: https://go.dev/blog/alias-names

[4] 

go doc: https://tonybai.com/2024/09/06/go-doc-add-http-support/

[5] 

Go 1.21: https://tonybai.com/2023/08/20/some-changes-in-go-1-21/

[6] 

实验特性Loopvar: https://go.dev/wiki/LoopvarExperiment

[7] 

Go 1.22版本: https://tonybai.com/2024/02/18/some-changes-in-go-1-22

[8] 

Arenas: https://github.com/golang/go/issues/51317

[9] 

Go 1.20版本: https://tonybai.com/2023/02/08/some-changes-in-go-1-20

[10] 

处于proposal hold状态: https://tonybai.com/2024/09/30/how-to-keep-up-with-go-evolution

[11] 

execution trace: https://tonybai.com/2021/06/28/understand-go-execution-tracer-by-example

[12] 

对泛型的支持: https://tonybai.com/2022/03/25/intro-generics

[13] 

panicnil: https://tonybai.com/2023/08/20/some-changes-in-go-1-21/

[14] 

Go, Backwards Compatibility, and GODEBUG: https://go.dev/doc/godebug

[15] 

Proposal: Extended backwards compatibility for Go: https://go.googlesource.com/proposal/+/master/design/56986-godebug.md

如果本文对你有所帮助,欢迎转发朋友圈!

点击下面标题,阅读更多干货!

Go 1.21中值得关注的几个变化

聊聊Go语言的向前兼容性和toolchain规则

Go 1.23引入的新Bug?其实只是文档没看仔细

Go 1.23中值得关注的几个变化

Go,14周年


Gopher部落知识星球在2024年将继续致力于打造一个高品质的Go语言学习和交流平台。我们将继续提供优质的Go技术文章首发和阅读体验。同时,我们也会加强代码质量和最佳实践的分享,包括如何编写简洁、可读、可测试的Go代码。此外,我们还会加强星友之间的交流和互动。欢迎大家踊跃提问,分享心得,讨论技术。我会在第一时间进行解答和交流。我衷心希望Gopher部落可以成为大家学习、进步、交流的港湾。让我相聚在Gopher部落,享受coding的快乐! 欢迎大家踊跃加入!

dac377450a18483644ebd9f1f095deb0.jpeg71b38ea9b1ee6feff9ac1666a998fa06.png

bf3a7b31b7ff38ed14ec0266cc83e3eb.png3044ad5dd8f22e2954f9931ffee72518.jpeg

著名云主机服务厂商DigitalOcean发布最新的主机计划,入门级Droplet配置升级为:1 core CPU、1G内存、25G高速SSD,价格5$/月。有使用DigitalOcean需求的朋友,可以打开这个链接地址:https://m.do.co/c/bff6eed92687 开启你的DO主机之路。

Gopher Daily(Gopher每日新闻) - https://gopherdaily.tonybai.com

我的联系方式:

  • 微博(暂不可用):https://weibo.com/bigwhite20xx

  • 微博2:https://weibo.com/u/6484441286

  • 博客:tonybai.com

  • github: https://github.com/bigwhite

  • Gopher Daily归档 - https://github.com/bigwhite/gopherdaily

  • Gopher Daily Feed订阅 - https://gopherdaily.tonybai.com/feed

c9253a2a4103519530e438b429c6bc04.jpeg

商务合作方式:撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值