Go 语言中数组函数的晚绑定:为什么所有函数都返回 5?
go 语言中的晚绑定
在编程中,我们经常需要处理数组中的函数。在 go 中,对数组中的函数进行晚绑定会带来一些有趣的情况。
import "fmt"type intorfunctype intfunc main() { var fns [5]intorfunctype for i := 0; i < 5; i++ { fns[i] = intorfunctype(func() int { return i }) } for n := 0; n < 5; n++ { fmt.printf("fns[%d]: %d", n, fns[n]()) }}
这段代码创建一个包含 5 个函数的数组。令人惊讶的是,输出的是:
fns[0]: 5fns[1]: 5fns[2]: 5fns[3]: 5fns[4]: 5
虽然我们希望输出的是 0 到 4,但为什么所有函数都返回 5?
这正是 go 语言中晚绑定的结果。在 go 中,代码在编译时不会执行函数,而是在运行时执行。这允许 go 语言通过函数指针和闭包等机制实现动态行为。
在我们的示例中,循环变量 i 在离开循环块后仍可访问,因为每个匿名函数都对其进行了闭包。这意味着所有函数都引用同一个 i,并且当它在循环的最后一次迭代中被更新为 5 时,所有函数都返回 5。
为了解决这个问题,我们可以采用闭包变量的闭区,如下所示:
for i := 0; i < 5; i++ { i := i // 创建一个闭区变量 fns[i] = IntOrFuncType(func() int { return i })}
现在,每个函数都具有自己的 i 变量并会返回正确的值。