PHP和Go中的闭包变量作用域

时间:2022-01-21 22:39:27

关于闭包函数,之前在聊过。这里忽略了一点,不管是Go/Php/Python,闭包都存在局部变量的引用。
我们还是先看个例子:

PHP示例:

$list = [];
for ($i = 0; $i < 3; $i++) {
    $list[] = function(){
        global $i;
        return $i;
    };
}
foreach ($list as $fun) {
    echo $fun().PHP_EOL;
}

以上输出:

3
3
3

相信很多朋友会说:不是应该输出 0 1 2吗?
好,继续,我们来看一个Golang的闭包示例:

package main
import "fmt"
func main() {
    var list []func() int
    for i := 0; i < 3; i++ {
        list = append(list, func() int {
            return i
        })
    }
    for _, fun := range list {
        fmt.Printf("%p = %v\n", fun, fun())
    }
}

以上输出:(这里输出的内存地址会根据机器的不同而有所不同,甚至相同的程序在不同的机器上执行后也会有不同的内存地址)

0x48fd70 = 3
0x48fd70 = 3
0x48fd70 = 3

有兴趣的博友可以写个同样的Python示例,相信结果同出一辙。那么为什么结果是 3?
重点:闭包函数在调用完毕后,i变量就是从循环条件中的 i++开始,也就是 原本是 0,1,2循环结束,那么闭包调用结束后此时 i从i++开始也就是3.
其实这个不难理解:
PHP:

$list = [];
for ($i = 0; $i < 3; $i++) {
    echo $i.PHP_EOL; // 输出 0 1 2
}
echo $i.PHP_EOL;// 输出 3

GO:

package main
import "fmt"
func main() {
    var i int
    for i = 0; i < 3; i++ {
        fmt.Printf("i = %v\n", i)
    }
    fmt.Printf("i = %v\n", i)
}
i = 0
i = 1
i = 2
i = 3

结论:循环结束后变量i 从i++开始。