首页 >后端开发 >Golang >为什么 Go 需要在闭包中显式复制循环变量?

为什么 Go 需要在闭包中显式复制循环变量?

Barbara Streisand
Barbara Streisand原创
2024-12-24 01:44:11353浏览

Why Does Go Require Explicit Copying of Loop Variables in Closures?

Go 中捕获的闭包(for 循环变量)

Go 编译器不会自动捕获 for...range 循环变量作为本地分配闭包变量。相反,Go 类似地对待所有 for 循环(包括 for...range 循环),需要将循环变量显式复制到局部闭包中以确保预期的行为。

复制循环变量的原因

此行为源于 Go 对 for 循环的一致处理。在所有类型的 for 循环中,包括 for...range 循环,循环变量的作用域为循环块。循环块结束后,循环变量将不再可访问。

在 for...range 循环中,循环变量在循环的每次迭代中都会被初始化为新值。但是,循环内创建的闭包继续引用原始循环变量。为了确保闭包具有循环变量的独立副本,必须将其分配给闭包内的局部变量。

代码示例

下面是一个示例这说明了复制循环变量的必要性:

func main() {
    m := make(map[int32]int32)
    for i := int32(1); i <= 10; i++ {
        m[i] = i
    }

    l := make([]func() (int32, int32), 0)
    for k, v := range m {
        l = append(l, func() (int32, int32) {
            return k, v
        })
    }

    for _, f := range l {
        k, v := f()
        fmt.Println(k, v)
    }
}

在此示例中,循环变量 k 预计在 for...range 的每次迭代期间发生变化 环形。然而,闭包捕获了原始循环变量 k,该变量在整个循环中保持不变。结果,代码打印同一对值 (10, 10) 十次。

解决方案:复制循环变量

要解决此问题,循环变量必须复制到闭包内的局部变量中:

func main() {
    m := make(map[int32]int32)
    for i := int32(1); i <= 10; i++ {
        m[i] = i
    }

    l := make([]func() (int32, int32), 0)
    for k, v := range m {
        kLocal, vLocal := k, v
        l = append(l, func() (int32, int32) {
            return kLocal, vLocal
        })
    }

    for _, f := range l {
        k, v := f()
        fmt.Println(k, v)
    }
}

现在,闭包捕获局部变量 kLocal 和vLocal,它独立地保存在 for...range 循环的每次迭代期间分配的值。代码正确打印了预期的值对 (1, 1), (2, 2), ..., (10, 10)。

以上是为什么 Go 需要在闭包中显式复制循环变量?的详细内容。更多信息请关注PHP中文网其他相关文章!

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