Go十大常见错误第7篇:不使用-race选项做并发竞争检测

2年前 (2022) 程序员胖胖胖虎阿
191 0 0

前言

这是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

推荐阅读

  • Go十大常见错误第1篇:未知枚举值
  • Go十大常见错误第2篇:benchmark性能测试的坑
  • Go十大常见错误第3篇:go指针的性能问题和内存逃逸
  • Go十大常见错误第4篇:break操作的注意事项
  • Go十大常见错误第5篇:Go语言Error管理
  • Go十大常见错误第6篇:slice初始化常犯的错误
  • Go面试题系列,看看你会几题?
  • Go编译器的race detector可以发现所有的并发冲突么?

开源地址

文章和示例代码开源在GitHub: Go语言初级、中级和高级教程。

公众号:coding进阶。关注公众号可以获取最新Go面试题和技术栈。

个人网站:Jincheng's Blog。

知乎:无忌。

福利

我为大家整理了一份后端开发学习资料礼包,包含编程语言入门到进阶知识(Go、C++、Python)、后端开发技术栈、面试题等。

关注公众号「coding进阶」,发送消息 backend 领取资料礼包,这份资料会不定期更新,加入我觉得有价值的资料。还可以发送消息「进群」,和同行一起交流学习,答疑解惑。

References

  • https://itnext.io/the-top-10-...
  • https://go.dev/doc/articles/r...
  • https://medium.com/@val_delep...

本文由mdnice多平台发布

相关文章

暂无评论

暂无评论...