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

时间:2021-08-22 15:02:05

关于闭包函数,之前在聊过。这里忽略了一点,不管是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;
}

以上输出:


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

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

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

0x48fd70 =
0x48fd70 =
0x48fd70 =

有兴趣的博友可以写个同样的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 = ; i < ; i++ {
fmt.Printf("i = %v\n", i)
}
fmt.Printf("i = %v\n", i)
}
i =
i =
i =
i =

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