首页  >  文章  >  后端开发  >  为什么在 Go 中共享变量会根据匿名函数中的范围导致不同的输出?

为什么在 Go 中共享变量会根据匿名函数中的范围导致不同的输出?

Barbara Streisand
Barbara Streisand原创
2024-11-07 02:40:02550浏览

Why does sharing variables in Go lead to different outputs depending on their scope within anonymous functions?

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中文网其他相关文章!

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn