【问题标题】:Range over string slice is inconsistent字符串切片的范围不一致
【发布时间】:2018-02-07 01:39:17
【问题描述】:

这段代码:

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)
}

产生输出“c c 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)
}

产生输出“[a b c]”

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

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

【问题讨论】:

标签: for-loop go closures goroutine for-range


【解决方案1】:

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

在此代码运行 3 次时:

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

将发送运行 3 次此代码:

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

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

解决方法是通过 func params 复制值:

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工作

【讨论】:

    猜你喜欢
    • 2019-06-06
    • 1970-01-01
    • 2019-12-30
    • 1970-01-01
    • 2013-11-23
    • 2020-11-27
    • 2020-12-01
    • 2012-03-18
    相关资源
    最近更新 更多