go語言採坑:閉包共享變數問題

tsin發表於2021-04-11

話不多說,直接看程式碼和註釋:

package main

import (
    "fmt"
    "time"
)

func main() {
    // 錯誤示例(列印出的結果是5個6)
    fmt.Println("closure buggy example...")
    for i := 1; i <= 5; i++ {
        // 每個goroutine共享一個變數,goroutine還沒開始的時候,i已經變成了6
        go func() {
            fmt.Println(i)
        }()
    }
    time.Sleep(1 * time.Second)


    // 正確示例1:
    fmt.Println("normal example...")
    for i := 1; i <= 5; i++ {
        go func(i int) {   // 使用區域性變數
            fmt.Println(i)
        }(i)
    }
    time.Sleep(1 * time.Second)

    // 正確示例2:
    fmt.Println("normal example 2...")
    for i := 1; i <= 5; i++ {
        i := i  // 為每個閉包建立一個變數
        go func() {
            fmt.Println(i)
        }()
    }
    time.Sleep(1 * time.Second)
}

執行得到的結果如下:

closure buggy example...
6
6
6
6
6
normal example...
5
2
1
3
4
normal example 2...
5
2
1
3
4

原因和解決方法見程式碼註釋。第一個例子是錯誤示例,第二、三個例子是正確示例。

本作品採用《CC 協議》,轉載必須註明作者和本文連結
Was mich nicht umbringt, macht mich stärker

相關文章