Go 中的数据竞争:为什么它发生在 10-11 毫秒以下?

Man*_*ath 3 go data-race

这是我运行的代码:

package main

import (
    "fmt"
    "time"
)

const delay = 9 * time.Millisecond

func main() {
    n := 0
    go func() {
        time.Sleep(delay)
        n++
    }()
    fmt.Println(n)
}
Run Code Online (Sandbox Code Playgroud)

这是我使用的命令:

go run -race data_race_demo.go
Run Code Online (Sandbox Code Playgroud)

这是我注意到的行为:

  • delay设定为9ms或更低,种族数据一直检测(程序引发Found 1 data race(s)
  • delay设定为12毫秒以上,种族数据从未检测(程序简单的打印0
  • delay设置为10至11毫秒,数据争用间断地出现(即,有时打印0有时抛出Found 1 data race(s)

为什么会在 10-11ms 左右发生这种情况?

darwin/amd64如果这很重要,我正在使用 Go 1.16.3 。

icz*_*cza 5

你有 2 个 goroutine:一个main和你启动的。他们在n没有同步的情况下访问变量(其中一个是写):这是数据竞争。

是否检测到这种竞争取决于是否发生这种竞争性访问。main()函数结束时,您的应用程序也会结束,它不会等待其他非maingoroutine 完成。

如果增加睡眠延迟,main()将比睡眠结束更早结束,并且不会等待n++活跃写入发生,因此不会检测到任何内容。如果 sleep 很短,比fmt.Prinln()执行时间短,则会发生并检测到异常写入。

没有什么关于10ms的特殊。这只是fmt.Println()您的环境中执行和终止您的应用程序所需的大致时间。如果您在Println()语句之前执行其他“冗长”的任务,例如:

for i := 0; i < 5_000_000_000; i++ {
}
fmt.Println(n)
Run Code Online (Sandbox Code Playgroud)

即使有 50 毫秒的睡眠时间也会检测到竞争(因为该循环需要一些时间来执行,允许在n读取fmt.Println()调用和应用程序终止之前发生不正常的写入)。(一个简单的time.Sleep()方法也可以,我只是不想让任何人得出错误的结论,即他们以某种方式相互“交互”。)

  • @ManuManjunath 10 毫秒没有什么特别的。这正是执行“fmt.Println()”并在“您的”环境中终止您的应用程序所需的时间。 (2认同)