如何理解围棋闭包的范围

有谁知道为什么case1输出相同的结果,而case2输出顺序结果?我知道case1输出相同值的原因是函数中每个函数的闭包将访问切片到相同的范围。


但是,为什么在每个循环中添加 i:=i 后,case2 可以输出顺序结果呢?在每循环中重新定义 i 后,是否会生成一个新作用域?喜欢让在爪哇脚本?


案例1


func main() {

    funcs := []func() {}


    for i:=0;i<10;i++{

        funcs = append(funcs, func() {

            fmt.Println(i)

        })

    }


    for i:=0;i<10;i++{

        funcs[i]()

    }

}

输出


10

10

10

10

10

10

10

10

10

10

案例2


func main() {

    funcs := []func() {}


    for i:=0;i<10;i++{

        i := i

        funcs = append(funcs, func() {

            fmt.Println(i)

        })

    }


    for i:=0;i<10;i++{

        funcs[i]()

    }

}

输出


0

1

2

3

4

5

6

7

8

9


www说
浏览 104回答 1
1回答

杨__羊羊

这并不是一个真正的作用域问题,而是编译器如何决定变量是通过引用还是按值捕获的问题。这并不总是显而易见的。对于 中的循环,在本例中重用迭代变量。在&nbsp;case1&nbsp;中,变量被捕获为引用,因为编译器在捕获发生后将其视为已更改。在情况 2&nbsp;中,内部变量是在捕获、捕获和释放之前创建的,因此编译器将其视为不变的,并按值捕获它。结果是,当函数在&nbsp;case1&nbsp;中运行时,结果都是最终值,因为这是变量最终的结果,并且每个函数仅包含对该特定 。在案例2中,每个捕获在创建时都传递了一个“副本”,因此它们显示了您期望的值。iiiiiii
打开App,查看更多内容
随时随地看视频慕课网APP

相关分类

Go