贝利信息

Golang 闭包的常见坑:for 循环 + 闭包 = 死亡组合

日期:2026-01-19 00:00 / 作者:舞夢輝影
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!
    }()
}

实际输出往往是:

3
3
3

因为所有闭包共享同一个 i 变量,循环结束时 i == 3,而 goroutine 是异步执行的,大概率在循环结束后才真正运行。

正确解法:让每个闭包拿到自己的值

不止 goroutine:defer、回调、切片遍历也中招

闭包陷阱不局限于 go 语句:

如何快速识别和规避