Go 语言史诗级更新-循环Bug修复

e07f0385e2996e56077f2c0bf1aebb97.png

背景

前两天 Golang 的官方博客更新了一篇文章:Fixing For Loops in Go 1.22

看这个标题的就是修复了 Go 循环的 bug,这真的是史诗级的更新;我身边接触到的大部分 Go 开发者都犯过这样的错误,包括我自己,所以前两年我也写过类似的博客:简单的 for 循环也会踩的坑

先来简单回顾下使用使用 for 循环会碰到的问题:

list := []*Demo{{"a"}, {"b"}}  
for _, v := range list {  
 go func() {  
  fmt.Println("name="+v.Name)  
 }()  
}  
  
type Demo struct {  
 Name string  
}

预期的结果应该是打印 a,b,但实际打印的却是b,b

a6feb71f3f816d9549df0e67f5b4a71c.png
image.png

Let's Encrypt: CAA Rechecking bug类似的问题连 mozilla 团队也没能幸免,所以也确实是一个非常常见的问题,这样的写法符合大部分的开发者的直觉,毕竟其他语言这么使用也没有问题。

当然在现阶段要解决也很简单,要么就是在使用之前先复制一次,或者使用闭包传参:

// 复制
 list := []*Demo{{"a"}, {"b"}}  
 for _, v := range list {  
  temp:=v  
  go func() {  
   fmt.Println("name="+temp.Name)  
  }()  
 }

 // 闭包
 list := []*Demo{{"a"}, {"b"}}  
 for _, v := range list {  
  go func(temp *Demo) {  
   fmt.Println("name="+temp.Name)  
  }(v)  
 }

还好官方也意识到了这个问题:0ef9d05cb803ca92897068f706c77102.png所以在 1.22 中我们可以不用再写这个 



v:=v这个多余的复制语句了,也不会出现上面的问题。

我们在 1.21 中可以使用环境变量预览这个特性:

❯ GOEXPERIMENT=loopvar go test
name=b
name=a

在 1.22 发布后建议大家都可以升级了,将这种恶心的 bug 扼杀在摇篮里。

1.22 后带来了一个好消息是今后少了一道面试题,坏消息是又新增了一个 1.22 版本带来了哪些变化的面试题😂

更多详情可以参看官方播客:https://go.dev/blog/loopvar-preview

往期推荐

Github的一个奇技淫巧

五分钟k8s实战-使用Ingress

k8s入门到实战--跨服务调用

k8s 入门到实战--部署应用到 k8s

0776e5a5a622fb00fdbc2c9ebc15089a.gif

点分享

5a45fe8c62c55e22bc6b054c1818adbd.gif

点点赞

5b9a8a712543d5d259d02e21ff13f113.gif

点在看

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值