Go中for循环变量复用导致闭包捕获同一地址:i从0到2迭代,但所有goroutine最终打印3(循环结束值),因闭包捕获的是i的引用而非每次迭代的值。
Go 语言中,for 循环里直接创建闭包并捕获循环变量,是引发“意料之外行为”的高频场景。根本原因在于:Go 的 for 循环变量是复用的——每次迭代都更新同一个变量的内存地址,而闭包捕获的是该变量的引用(地址),不是值。
以下代码本意是启动 3 个 goroutine,分别打印 0、1、2:
for i := 0; i < 3; i++ {
go func() {
fmt.Println(i) // 全部输出 3!
}()
}
实际输出往往是:

因为所有闭包共享同一个 i 变量,循环结束时 i == 3,而 goroutine 是异步执行的,大概率在循环结束后才真正运行。
for i := 0; i < 3; i++ {
go func(val int) {
fmt.Println(val)
}(i)
}:= 在循环体内创建新变量,绑定当前值for i := 0; i < 3; i++ {
i := i // 关键:遮蔽外层 i,创建新变量
go func() {
fmt.Println(i)
}()
}闭包陷阱不局限于 go 语句:
defer 中调用闭包:同样会延迟到函数返回时执行,看到的是最终的循环变量值range 遍历切片/映射时,key 和 value 也是复用变量for k, v := range m {
fns = append(fns, func() { fmt.Printf("%s: %d\n", k, v) })
}k 和 v
i、k、v 等),就危险staticcheck 能报出 SA5008(loop variable captured by func literal)警告