
go 语言中的晚绑定
在编程中,我们经常需要处理数组中的函数。在 go 中,对数组中的函数进行晚绑定会带来一些有趣的情况。
import "fmt"
type intorfunctype int
func 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", n, fns[n]())
}
}这段代码创建一个包含 5 个函数的数组。令人惊讶的是,输出的是:
fns[0]: 5 fns[1]: 5 fns[2]: 5 fns[3]: 5 fns[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 变量并会返回正确的值。










