Goroutine 循环变量陷阱

作者:matrix 发布时间:2025-01-31 分类:Golang

很经典的Golang问题,稍不注意处理这就是个坑

错误case

package main

import (
    "fmt"
    "time"
)

func main() {
    for i := 0; i < 5; i++ {
        go func() {
            fmt.Println(i) // 捕获了变量 i 的地址
        }()
    }
    time.Sleep(1 * time.Second) // 等待所有 goroutine 执行完毕
}

这是由于 Go 的闭包行为引起的:闭包会捕获变量的地址,而不是其值。

解决方案

1. 传递参数到闭包中

package main

import (
    "fmt"
    "time"
)

func main() {
    for i := 0; i < 5; i++ {
        go func(n int) { // 使用参数 n
            fmt.Println(n)
        }(i) // 将 i 作为参数传递
    }
    time.Sleep(1 * time.Second) // 等待所有 goroutine 执行完毕
}

2. 在循环内部创建局部变量

在循环体内创建一个新的局部变量,使每个 goroutine 捕获独立的变量。

package main

import (
    "fmt"
    "time"
)

func main() {
    for i := 0; i < 5; i++ {
        i := i // 创建一个新的局部变量
        go func() {
            fmt.Println(i)
        }()
    }
    time.Sleep(1 * time.Second) // 等待所有 goroutine 执行完毕
}

新春快乐🎉🎉 PEACE~~