这是Go十大常见错误系列的第7篇:不使用-race
选项做并发竞争检测。素材来源于Go布道者,现Docker公司资深工程师Teiva Harsanyi。
本文涉及的源代码全部开源在:Go十大常见错误源代码,欢迎大家关注公众号,及时获取本系列最新更新。
并发编程里很容易遇到并发访问冲突的问题。
Go语言里多个goroutine同时操作某个共享变量的时候,如果一个goroutine对该变量做写操作,其它goroutine做读操作,假设没有做好并发访问控制,就容易出现并发访问冲突,导致程序crash。
大家可以看如下的代码示例:
package main
import (
"fmt"
)
func main() {
c := make(chan bool)
m := make(map[string]string)
go func() {
m["1"] = "a" // First conflicting access.
c <- true
}()
m["2"] = "b" // Second conflicting access.
<-c
for k, v := range m {
fmt.Println(k, v)
}
}
上面的代码会出现并发访问冲突,2个goroutine同时对共享变量m
做写操作。
通过-race
选项,我们就可以利用编译器帮我们快速发现问题。
$ go run -race main.go
==================
WARNING: DATA RACE
Write at 0x00c000074180 by goroutine 7:
runtime.mapassign_faststr()
/usr/local/opt/go/libexec/src/runtime/map_faststr.go:202 +0x0
main.main.func1()
/Users/xxx/github/go-tutorial/workspace/senior/p28/data-race/main.go:11 +0x5d
Previous write at 0x00c000074180 by main goroutine:
runtime.mapassign_faststr()
/usr/local/opt/go/libexec/src/runtime/map_faststr.go:202 +0x0
main.main()
/Users/xxx/github/go-tutorial/workspace/senior/p28/data-race/main.go:14 +0xcb
Goroutine 7 (running) created at:
main.main()
/Users/xxx/github/go-tutorial/workspace/senior/p28/data-race/main.go:10 +0x9c
==================
1 a
2 b
Found 1 data race(s)
exit status 66
一个常见的错误是开发者测试Go程序的时候,不使用-race
选项。
尽管Go语言设计的目的之一是为了让并发编程更简单、更不容易出错,但Go语言开发者还是会遇到并发问题。
因此,大家在测试Go程序的时候,应该开启-race
选项,及时发现代码里的并发访问冲突问题。
$ go test -race mypkg // to test the package
$ go run -race mysrc.go // to run the source file
$ go build -race mycmd // to build the command
$ go install -race mypkg // to install the package
文章和示例代码开源在GitHub: Go语言初级、中级和高级教程。
公众号:coding进阶。关注公众号可以获取最新Go面试题和技术栈。
个人网站:Jincheng's Blog。
知乎:无忌。
我为大家整理了一份后端开发学习资料礼包,包含编程语言入门到进阶知识(Go、C++、Python)、后端开发技术栈、面试题等。
关注公众号「coding进阶」,发送消息 backend 领取资料礼包,这份资料会不定期更新,加入我觉得有价值的资料。还可以发送消息「进群」,和同行一起交流学习,答疑解惑。