Go 中共享变量:了解两种情况的区别
在 Go 中,goroutines 是轻量级的并发任务,可以共享内存并使用以下方式进行通信渠道。 Goroutines 之间的变量共享需要小心处理,以避免数据竞争问题。
考虑以下 Go 程序:
<code class="go">package main import ( "fmt" "sync" ) func main() { var wg sync.WaitGroup for i := 0; i < 5; i++ { wg.Add(1) x := i go func() { defer wg.Done() fmt.Println(x) }() } wg.Wait() fmt.Println("Done") }</code>
执行时,该程序会打印预期的输出:
4 0 1 3 2
在这种情况下,每个 goroutine 都有自己的变量 x 副本,该副本在创建 goroutine 时使用 i 的当前值进行初始化。这是因为 x 是在匿名函数中声明的,并且其范围仅限于该函数。
现在,考虑对程序进行轻微修改:
<code class="go">package main import ( "fmt" "sync" ) func main() { var wg sync.WaitGroup for i := 0; i < 5; i++ { wg.Add(1) go func() { defer wg.Done() fmt.Println(i) }() } wg.Wait() fmt.Println("Done") }</code>
这一次,输出变成:
5 5 5 5 5
解释在于变量 i 在 goroutine 中的使用方式。由于 i 是在匿名函数外部声明的,因此它在所有 goroutine 之间共享。在这种情况下,当每个 goroutine 执行 fmt.Println(i) 时,它会打印 i 的最终值,即 5。
为了验证这一点,我们可以在其中添加 x 和 i 的内存地址的打印goroutine。输出显示 x 对于每个 goroutine 有不同的地址,而 i 对于所有 goroutine 有相同的地址:
0xc0420301e0 0xc0420301f8 0xc0420301e8 0xc0420301f0 0xc042030200 0xc042030208
综上所述,两种情况之间变量共享的差异源于变量的范围在匿名函数中声明。当一个变量在匿名函数中声明时,它对于该 goroutine 是私有的。另一方面,在匿名函数外部声明的变量在所有 goroutine 之间共享。
以上是为什么在 Go 中共享变量会根据匿名函数中的范围导致不同的输出?的详细内容。更多信息请关注PHP中文网其他相关文章!