首页 > 文章列表 > Go 语言中是否存在竞争条件?

Go 语言中是否存在竞争条件?

226 2024-02-27
问题内容

func main() {
    m := map[string]int{
        "foo": 42,
        "bar": 1337,
    }

    go func() {
        time.Sleep(1 * time.Second)
        tmp := map[string]int{
            "foo": 44,
            "bar": 1339,
        }

        m = tmp
    }()

    for {
        val := m["foo"]
        fmt.Println(val)
    }
}

我在很多包中都看到了这个。

为什么这不被视为竞争条件?

go run -race . 没有错误。


正确答案


正如@volker 所指出的,这是一场数据竞赛。而且由于只有一次写入,因此很难被检测到。这是一个修改后的演示,可以轻松触发数据争用错误:

package main

import (
    "fmt"
    "time"
)

func main() {
    m := map[string]int{
        "foo": 42,
        "bar": 1337,
    }

    done := make(chan any)

    go func() {
        for i := 0; i < 100; i++ {
            time.Sleep(time.Microsecond)
            tmp := map[string]int{
                "foo": 44,
                "bar": 1339,
            }

            m = tmp
        }

        close(done)
    }()

    for {
        select {
        case <-done:
            return
        default:
            val := m["foo"]
            fmt.Println(val)
        }
    }
}