快看! Go 1.22 对for循环进行了两个大更新

共 4730字,需浏览 10分钟

 ·

2024-04-11 11:30


前言


Go 1.22 版本于 2024 年 2 月 6 日正式向世界宣告了版本的发布 。


我们可以从官网下载1.22版本进行体验,或者从 Go Playground上进行体验最新语法


8dd1c510c96e36bbb8d294b367012de3.webp

值得注意的是在语言层面上,这个版本对 for 循环进行了两处更新:



  • • for循环的每次迭代都会定义新变量,而不再是共享一个变量


  • • 支持对整数范围进行循环迭代



今天将以案例的方式对比下最新版本 for 循环的两个更新点。


🧐 Let's Go!


循环不再共享循环变量



🔔 for在循环语义层面的坑



Go1.22之前版本for 循环声明的变量只创建一次,并在每次迭代中进行更新,这会导致遍历时访问value时实际上都是访问的同一个地址的值。


相信不少小伙伴都遇到过,特别是在初学Go的时候!


Go1.22之前版本


我们用官博文章中那个例子,稍微改进如下,并使用1.21版本运行


    

package main

import "fmt"

func main() {
    done := make(chan bool)

    values := []string{"xiao", "xu", "code"}
    for _, v := range values {
        go func() {
            fmt.Println(v)
            done <- true
        }()
    }

    // 等待所有的 goroutine 执行结束
    for _ = range values {
        <-done
    }
}



上述代码运行结果如下所示:


    

code
code
code



这三个创建的 goroutine 都在打印同一个变量 v,所以它们通常会打印出 "code"、"code"、"code",而不是以某种顺序打印出 "xiao"、"xu" 和 "code"。



🚩 这就是共享循环变量造成的问题!



这个比较好理解,这个循环的 v 只创建一次,在每次循环的时候都会更新,而 闭包在访问 v 时实际上都访问的是同一个内存地址,所以最终打印的都是同一个值。


解决办法:


在Go版本不变的情况下,可以通过下面两种方式修改代码避免这个问题。


1:将for循环中传入v,代码改造如下


    

values := []string{"xiao", "xu", "code"}
for _, v := range values {
    go func(v string) {
        fmt.Println( v)
        done <- true
    }(v)
}



2:在循环中重新定义一个变量进行再次赋值


    

values := []string{"xiao", "xu", "code"}
for _, v := range values {
    value := v
    go func() {
        fmt.Println( value)
        done <- true
    }()
}



Go1.22版本


不过这个问题在1.22版本已经得到处理了,大家用这个版本的时候可以放心使用了,太爽了吧!


我们在1.22版本上运行和1.21一样的代码


    

package main

import "fmt"

func main() {
    done := make(chan bool)

    values := []string{"xiao", "xu", "code"}
    for _, v := range values {
        go func() {
            fmt.Println(v)
            done <- true
        }()
    }

    // 等待所有的 goroutine 执行结束
    for _ = range values {
        <-done
    }
}



上述代码运行结果如下所示:


    

code
xiao
xu



for 循环的每次迭代都会创建新变量,每次循环迭代各自的变量,以避免意外共享错误。上面一模一样的代码,输出结果不再是固定的 code。


支持整数范围进行循环迭代


在 Go 1.22 版本之前, for range 仅支持对 array or slice、string、map 和 channel 类型的进行迭代。


而自 Go 1.22 版本起,新增了整数类型的迭代支持,我们能够直接使用整数进行循环迭代。


下面同样列举不同版本的例子,看看差异性!


Go1.22之前版本


    

package main

import "fmt"

func main() {
    for i := range 5 {
        fmt.Println("小许code", i)
    }
}



不支持遍历整数范围,这个range 5就直接提示报错了,编译当然有问题了


    

.\main.go:15:17: cannot range over 5 (untyped int constant)


Go1.22版本


    

package main

import "fmt"

func main() {
    for i := range 5 {
        fmt.Println("小许code", i)
    }
}



上述代码运行结果如下所示:


    
小许code 0
小许code 1
小许code 2
小许code 3
小许code 4



今天关于Go 1.22关于for循环的更新就介绍到这了,是不是觉得这个更新太棒啦!


注意了,面试的同学,如果没有指定说明版本的话,还是需要注意下调整回之前的答案!






推荐阅读




福利

我为大家整理了一份 从入门到进阶的Go学习资料礼包 ,包含学习建议:入门看什么,进阶看什么。 关注公众号 「polarisxu」,回复  ebook  获取;还可以回复「进群」,和数万 Gopher 交流学习。


浏览 57
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报