-2

这段代码:

import "fmt"
import "time"
func main() {
    string_slice:=[]string{"a","b","c"}

    for _,s:=range string_slice{
        go func(){
            time.Sleep(1*time.Second)
            fmt.Println(s)
        }()
    }

    time.Sleep(3*time.Second)
}

产生输出“cc c”,而这段代码:

import "fmt"
func main() {
    string_slice:=[]string{"a","b","c"}

    for _,s:=range string_slice{
        s="asd"
        fmt.Println(s)
    }
    fmt.Println(string_slice)
}

产生输出“[abc]”

第一个建议 for range 迭代引用(它不应该),第二个建议它迭代值的副本(它应该)。

为什么第一个不产生输出“ab c”?

4

1 回答 1

0

当使用 goroutine 时,你必须假设它会并行运行。所以在这种情况下可能会出现 'cc c' 和 'bb b' 或 'aa a'

运行 3 次此代码:

for _,s:=range string_slice \\run t0, t1, t2 

将发送运行3 次此代码:

go func(){
  fmt.Println(s)
}()//send in t0, t1, t2

因此,例如,func() 可能会在 t2 开始运行。在这种情况下,结果将是“cc c”,因为s等于最新值 (string_slice[2])。

解决方案是通过 func 参数复制值:

for _, s := range string_slice{
    go func(x string){
        time.Sleep(1*time.Second)
        fmt.Println(x)
    }(s)
}

或每次迭代创造新价值

//creating new value per iteration
for i := range string_slice{
    s := string_slice[i]
    go func(){
        time.Sleep(1*time.Second)
        fmt.Println(s)
    }()
}

请参阅https://play.golang.org/p/uRD6Qy6xSw中的工作

于 2017-08-29T15:19:40.013 回答