sync.Mutex

我们已经看到信道非常适合在各个 Go 程间进行通信。

但是如果我们并不需要通信呢?比如说,若我们只是想保证每次只有一个 Go 程能够访问一个共享的变量,从而避免冲突?

这里涉及的概念叫做 互斥(mutualexclusion) ,我们通常使用 互斥锁(Mutex)* 这一数据结构来提供这种机制。

Go 标准库中提供了 sync.Mutex 互斥锁类型及其两个方法:

  • Lock
  • Unlock

    我们可以通过在代码前调用 Lock 方法,在代码后调用 Unlock 方法来保证一段代码的互斥执行。参见 Inc 方法。

    我们也可以用 defer 语句来保证互斥锁一定会被解锁。参见 Value 方法。

mutex-counter.go

  1. package main
  2. import (
  3. "fmt"
  4. "sync"
  5. "time"
  6. )
  7. // SafeCounter 的并发使用是安全的。
  8. type SafeCounter struct {
  9. v map[string]int
  10. mux sync.Mutex
  11. }
  12. // Inc 增加给定 key 的计数器的值。
  13. func (c *SafeCounter) Inc(key string) {
  14. c.mux.Lock()
  15. // Lock 之后同一时刻只有一个 goroutine 能访问 c.v
  16. c.v[key]++
  17. c.mux.Unlock()
  18. }
  19. // Value 返回给定 key 的计数器的当前值。
  20. func (c *SafeCounter) Value(key string) int {
  21. c.mux.Lock()
  22. // Lock 之后同一时刻只有一个 goroutine 能访问 c.v
  23. defer c.mux.Unlock()
  24. return c.v[key]
  25. }
  26. func main() {
  27. c := SafeCounter{v: make(map[string]int)}
  28. for i := 0; i < 1000; i++ {
  29. go c.Inc("somekey")
  30. }
  31. time.Sleep(time.Second)
  32. fmt.Println(c.Value("somekey"))
  33. }