我报名参加金石计划1期挑战——瓜分10万奖池,这是我的第2篇文章,点击查看活动详情
前言
这是Go十大常见错误系列的第10篇:Goroutine和循环变量一起使用的坑。素材来源于Go布道者,现Docker公司资深工程师Teiva Harsanyi。
本文涉及的源代码全部开源在:Go十大常见错误源代码,欢迎大家关注,及时获取本系列最新更新。
常见错误
对于Go初学者,很容易犯的一个错误就是goroutine和循环变量结合在一起使用时,错误地使用了循环变量。
比如下面这个例子:
ints := []int{1, 2, 3}
for _, i := range ints {
go func() {
fmt.Printf("%v\n", i)
}()
}
这段程序的输出结果应该是什么?
Go初学者可能认为输出结果应该是1 2 3,但实际情况并不是。
这个例子里,3个goroutine共享同一个变量i,最后输出的结果大概率是输出3 3 3。
要解决这个问题,主要有2个解决方案。
解决方案1
把循环变量i作为goroutine函数的一个参数,编译器在执行go func(i int)时,就会解析到i的值,确保每个goroutine可以拿到自己想要的值。
ints := []int{1, 2, 3}
for _, i := range ints {
go func(i int) {
fmt.Printf("%v\n", i)
}(i)
}
解决方案2
创建一个新的变量,用于goroutine。
ints := []int{1, 2, 3}
for _, i := range ints {
i := i
go func() {
fmt.Printf("%v\n", i)
}()
}
推荐阅读
- Go十大常见错误第1篇:未知枚举值
- Go十大常见错误第2篇:benchmark性能测试的坑
- Go十大常见错误第3篇:go指针的性能问题和内存逃逸
- Go十大常见错误第4篇:break操作的注意事项
- Go十大常见错误第5篇:Go语言Error管理
- Go十大常见错误第6篇:slice初始化常犯的错误
- Go十大常见错误第7篇:不使用-race选项做并发竞争检测
- Go十大常见错误第8篇:并发编程中Context使用常见错误
- Go面试题系列,看看你会几题?
开源地址
文章和示例代码开源在GitHub: Go语言初级、中级和高级教程。
公众号:coding进阶。
个人网站:Jincheng's Blog。
知乎:无忌。