Go中使用for..range语句时需要注意的地方

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

你可能感兴趣的:(学习GO语言)