for..range 循环时需要特别注意:
var s []int = []int{1, 2, 3}
for _, value := range s {
t.Logf("the value address is %p", &value)
}
执行上面的代码,输出结果如下:
blog_test.go:8: the value address is 0xc0420f2df8
blog_test.go:8: the value address is 0xc0420f2df8
blog_test.go:8: the value address is 0xc0420f2df8
可以看出value指向相同的地址。所以在for循环中使用并发就需要特别注意。因为并发,不同的goroutine可能会读取到相同的值。
简单修改一下上面的例子,将打印语句修改为并发打印。
var wg sync.WaitGroup
var s []int = []int{1, 2, 3}
for _, value := range s {
wg.Add(1)
go func(){
defer wg.Done()
t.Logf("go routine value %d", value)
}()
}
wg.Wait()
执行结果如下,,发现所有的值都是3.
blog_test.go:15: go routine value 3
blog_test.go:15: go routine value 3
blog_test.go:15: go routine value 3
如何解决上面的问题,需要使用到值传递,重新再匿名函数中声明一个参数。
var wg sync.WaitGroup
var s []int = []int{1, 2, 3}
for _, value := range s {
wg.Add(1)
go func(value int) {
defer wg.Done()
t.Logf("go routine value %d", value)
}(value)
}
wg.Wait()
执行结果就正常了:
blog_test.go:15: go routine value 3
blog_test.go:15: go routine value 2
blog_test.go:15: go routine value 1