




go test -race 是官方唯一推荐的竞态检测方式,通过运行时插桩监控内存读写,需配合真实并发测试触发,命令顺序必须正确,修复后须再次验证零警告。
别信静态分析或“看代码觉得没问题”——Go 的竞态(data race)只在真实并发执行时暴露,go test -race 是官方唯一推荐、开箱即用的运行时检测手段。它不是预测,而是插桩监控:每个内存读写都被记录,一旦发现“goroutine A 写、goroutine B 读/写同一地址且无同步”,立刻报错。
go run -race main.go 很难复现,因为启动时机、调度不可控;而 go test -race 可精确控制 goroutine 数量、启动节奏和共享变量访问频率Inc(),而不是只启 2 个 goroutine 跑一次c == 1000 通过 ≠ 没竞态;只有 go test -race 不报警,才算过关-race 必须紧跟在主命令之后、包路径之前,顺序错就静默失效。常见错误写法:go run main.go -race(-race 被当成了程序参数)、go test ./ -race(位置靠后,Go 忽略)。
go test -race ./(推荐,覆盖整个模块),go test -race -v ./(加 -v 看详细日志),go test -race pkgname(指定包)go build -race -o app .,但注意它只支持 amd64 和 arm64,交叉编译到 32 位会失败go test -race ./,避免漏掉新引入的竞态报错不是堆栈异常,而是一组“冲突快照”:两个 goroutine 在相近时间访问了同一内存地址,检测器记录下它们各自的调用栈。关键不是“谁先谁后”,而是“没同步”。
Write at 0x00c0000a0060 by goroutine 7 和 Previous read at 0x00c0000a0060 by goroutine 6 表明是同一个变量地址被并发读写main.go:12,直接跳转就能看到问题语句,比如 counter++ 或 m["key"] = val
Previous write 当成时间先后——竞态的本质就是顺序不确定,检测器只是按自己记录顺序命名而已加了 sync.Mutex、改用 atomic.AddInt64 或换成 chan,不等于问题消失。锁

go test -race,必须零警告for i := range xs { go func() { use(i) }() } 会导致所有 goroutine 共享最后一个 i 值,-race 通常能抓到这类读写冲突Get/Put 自身线程安全,如果放进去的对象内部有可变状态,仍需额外同步go test -race 不报警,才是唯一的验收标准。