Even with the greatest of care, it’s all too easy to make concurrency mistakes. Fortunately, the Go runtime and toolchain are equipped with a sophisticated and easy-to-use dynamic analysis tool, the race detector.
Just add the -race
flag to your go build
, go run
,
or go test
command.
This causes the compiler to build a modified version of your
application or test with additional instrumentation
that effectively records all accesses to shared variables
that occurred during execution, along with the identity of the
goroutine that read or wrote the variable.
In addition, the modified program records all synchronization events,
such as go
statements, channel operations, and calls to
(*sync.Mutex).Lock
, (*sync.WaitGroup).Wait
, and so on.
(The complete set of synchronization events is specified by the The
Go Memory Model document that accompanies the language specification.)
The race detector studies this stream of events, looking for cases in which one goroutine reads or writes a shared variable that was most recently written by a different goroutine without an intervening synchronization operation. This indicates a concurrent access to the shared variable, and thus a data race. The tool prints a report that includes the identity of the variable, and the stacks of active function calls in the reading goroutine and the writing goroutine. This is usually sufficient to pinpoint the problem. Section 9.7 contains an example of the race detector in action.
The race detector reports all data races that were actually executed. However, it can only detect race conditions that occur during a run; it cannot prove that none will ever occur. For best results, make sure that your tests exercise your packages using concurrency.
Due to extra bookkeeping, a program built with race detection needs more time and memory to run, but the overhead is tolerable even for many production jobs. For infrequently occurring race conditions, letting the race detector do its job can save hours or days of debugging.
3.141.199.243