如何使用Mutex确保并发程序的正确性

打印 上一主题 下一主题

主题 546|帖子 546|积分 1638

1. 简介

本文的主要内容是介绍Go中Mutex并发原语。包含Mutex的基本使用,使用的注意事项以及一些实践建议。
2. 基本使用

2.1 基本定义

Mutex是Go语言中的一种同步原语,全称为Mutual Exclusion,即互斥锁。它可以在并发编程中实现对共享资源的互斥访问,保证同一时刻只有一个协程可以访问共享资源。Mutex通常用于控制对临界区的访问,以避免竞态条件的出现。
2.2 使用方式

使用Mutex的基本方法非常简单,可以通过调用Mutex的Lock方法来获取锁,然后通过Unlock方法释放锁,示例代码如下:
  1. import "sync"
  2. var mutex sync.Mutex
  3. func main() {
  4.   mutex.Lock()    // 获取锁
  5.   // 执行需要同步的操作
  6.   mutex.Unlock()  // 释放锁
  7. }
复制代码
2.3 使用例子

2.3.1 未使用mutex同步代码示例

下面是一个使用goroutine访问共享资源,但没有使用Mutex进行同步的代码示例:
  1. package main
  2. import (
  3.     "fmt"
  4.     "time"
  5. )
  6. var count int
  7. func main() {
  8.     for i := 0; i < 1000; i++ {
  9.         go add()
  10.     }
  11.     time.Sleep(1 * time.Second)
  12.     fmt.Println("count:", count)
  13. }
  14. func add() {
  15.     count++
  16. }
复制代码
上述代码中,我们启动了1000个goroutine,每个goroutine都调用add()函数将count变量的值加1。由于count变量是共享资源,因此在多个goroutine同时访问的情况下会出现竞态条件。但是由于没有使用Mutex进行同步,所以会导致count的值无法正确累加,最终输出的结果也会出现错误。
在这个例子中,由于多个goroutine同时访问count变量,而不进行同步控制,导致每个goroutine都可能读取到同样的count值,进行相同的累加操作。这就会导致最终输出的count值不是期望的结果。如果我们使用Mutex进行同步控制,就可以避免这种竞态条件的出现。
2.3.2 使用mutex解决上述问题

下面是使用Mutex进行同步控制,解决上述代码中竞态条件问题的示例:
  1. package main
  2. import (
  3.     "fmt"
  4.     "sync"
  5.     "time"
  6. )
  7. var (
  8.     count int
  9.     mutex sync.Mutex
  10. )
  11. func main() {
  12.     for i := 0; i < 1000; i++ {
  13.         go add()
  14.     }
  15.     time.Sleep(1 * time.Second)
  16.     fmt.Println("count:", count)
  17. }
  18. func add() {
  19.     mutex.Lock()
  20.     count++
  21.     mutex.Unlock()
  22. }
复制代码
在上述代码中,我们在全局定义了一个sync.Mutex类型的变量mutex,用于进行同步控制。在add()函数中,我们首先调用mutex.Lock()方法获取mutex的锁,确保只有一个goroutine可以访问count变量。然后进行加1操作,最后调用mutex.Unlock()方法释放mutex的锁,使其他goroutine可以继续访问count变量。
通过使用Mutex进行同步控制,我们避免了竞态条件的出现,确保了count变量的正确累加。最终输出的结果也符合预期。
3. 使用注意事项

3.1 Lock/Unlock需要成对出现

下面是一个没有成对出现Lock和Unlock的代码例子:
  1. package main
  2. import (
  3.     "fmt"
  4.     "sync"
  5. )
  6. func main() {
  7.     var mutex sync.Mutex
  8.     go func() {
  9.         mutex.Lock()
  10.         fmt.Println("goroutine1 locked the mutex")
  11.     }()
  12.     go func() {
  13.         fmt.Println("goroutine2 trying to lock the mutex")
  14.         mutex.Lock()
  15.         fmt.Println("goroutine2 locked the mutex")
  16.     }()
  17. }
复制代码
在上述代码中,我们创建了一个sync.Mutex类型的变量mutex,然后在两个goroutine中使用了这个mutex。
在第一个goroutine中,我们调用了mutex.Lock()方法获取mutex的锁,但是没有调用相应的Unlock方法。在第二个goroutine中,我们首先打印了一条信息,然后调用了mutex.Lock()方法尝试获取mutex的锁。由于第一个goroutine没有释放mutex的锁,第二个goroutine就一直阻塞在Lock方法中,一直无法执行。
因此,在使用Mutex的过程中,一定要确保每个Lock方法都有对应的Unlock方法,确保Mutex的正常使用。
3.2 不能对已使用的Mutex作为参数进行传递

下面举一个已使用的Mutex作为参数进行传递的代码的例子:
  1. type Counter struct {
  2.     sync.Mutex
  3.     Count int
  4. }
  5. func main(){
  6.     var c Counter
  7.     c.Lock()
  8.     defer c.Unlock()
  9.     c.Count++
  10.     foo(c)
  11.     fmt.println("done")
  12. }
  13. func foo(c Counter) {
  14.     c.Lock()
  15.     defer c.Unlock()
  16.     fmt.println("foo done")
  17. }
复制代码
当一个 mutex 被传递给一个函数时,预期的行为应该是该函数在访问受 mutex 保护的共享资源时,能够正确地获取和释放 mutex,以避免竞态条件的发生。
如果我们在Mutex未解锁的情况下拷贝这个Mutex,就会导致锁失效的问题。因为Mutex的状态信息被拷贝了,拷贝出来的Mutex还是处于锁定的状态。而在函数中,当要访问临界区数据时,首先肯定是先调用Mutex.Lock方法加锁,而传入Mutex其实是处于锁定状态的,此时函数将永远无法获取到锁。
因此,不能将已使用的Mutex直接作为参数进行传递。
3.3 不可重复调用Lock/UnLock方法

下面是一个例子,其中对同一个 Mutex 进行了重复加锁:
  1. package main
  2. import (
  3.     "fmt"
  4.     "sync"
  5. )
  6. func main() {
  7.     var mu sync.Mutex
  8.     mu.Lock()
  9.     fmt.Println("First Lock")
  10.     // 重复加锁
  11.     mu.Lock()
  12.     fmt.Println("Second Lock")
  13.     mu.Unlock()
  14.     mu.Unlock()
  15. }
复制代码
在这个例子中,我们先对 Mutex 进行了一次加锁,然后在没有解锁的情况下,又进行了一次加锁操作.
这种情况下,程序会出现死锁,因为第二次加锁操作已经被阻塞,等待第一次加锁的解锁操作,而第一次加锁的解锁操作也被阻塞,等待第二次加锁的解锁操作,导致了互相等待的局面,无法继续执行下去。
Mutex实际上是通过一个int32类型的标志位来实现的。当这个标志位为0时,表示这个Mutex当前没有被任何goroutine获取;当标志位为1时,表示这个Mutex当前已经被某个goroutine获取了。
Mutex的Lock方法实际上就是将这个标志位从0改为1,表示获取了锁;Unlock方法则是将标志位从1改为0,表示释放了锁。当第二次调用Lock方法,此时标记位为1,代表有一个goroutine持有了这个锁,此时将会被阻塞,而持有该锁的其实就是当前的goroutine,此时该程序将会永远阻塞下去。
4. 实践建议

4.1 Mutex锁不要同时保护两份不相关数据

下面是一个例子,使用Mutex同时保护两份不相关的数据
  1. // net/http transport.go
  2. type Transport struct {
  3.    lk       sync.Mutex
  4.    idleConn map[string][]*persistConn
  5.    altProto map[string]RoundTripper // nil or map of URI scheme => RoundTripper
  6. }
  7. func (t *Transport) CloseIdleConnections() {
  8.    t.lk.Lock()
  9.    defer t.lk.Unlock()
  10.    if t.idleConn == nil {
  11.       return
  12.    }
  13.    for _, conns := range t.idleConn {
  14.       for _, pconn := range conns {
  15.          pconn.close()
  16.       }
  17.    }
  18.    t.idleConn = nil
  19. }
  20. func (t *Transport) RegisterProtocol(scheme string, rt RoundTripper) {
  21.    if scheme == "http" || scheme == "https" {
  22.       panic("protocol " + scheme + " already registered")
  23.    }
  24.    t.lk.Lock()
  25.    defer t.lk.Unlock()
  26.    if t.altProto == nil {
  27.       t.altProto = make(map[string]RoundTripper)
  28.    }
  29.    if _, exists := t.altProto[scheme]; exists {
  30.       panic("protocol " + scheme + " already registered")
  31.    }
  32.    t.altProto[scheme] = rt
  33. }
复制代码
在这个例子中,idleConn是存储了空闲的连接,altProto是存储了协议的处理器,CloseIdleConnections方法是关闭所有空闲的连接,RegisterProtocol是用于注册协议处理的。
尽管ideConn和altProto这两部分数据并没有任何关联,但是却是使用同一个Mutex来保护的,这样子当调用RegisterProtocol方法时,便无法调用CloseIdleConnections方法,这会导致竞争过多,从而影响性能。
因此,为了提高并发性能,应该将 Mutex 的锁粒度尽量缩小,只保护需要保护的数据。
现代版本的 net/http 中已经对 Transport 进行了改进,分别使用了不同的 mutex 来保护 idleConn 和 altProto,以提高性能和代码的可维护性。
  1. type Transport struct {
  2.    idleMu       sync.Mutex
  3.    idleConn     map[connectMethodKey][]*persistConn // most recently used at end
  4.    altMu    sync.Mutex   // guards changing altProto only
  5.    altProto atomic.Value // of nil or map[string]RoundTripper, key is URI scheme   
  6. }
复制代码
4.2 Mutex嵌入结构体中位置放置建议

将 Mutex 嵌入到结构体中,如果只需要保护其中一些数据,可以将 Mutex 放在需要控制的字段上面,然后使用空格将被保护字段和其他字段进行分隔。这样可以实现更细粒度的锁定,也能更清晰地表达每个字段需要被互斥保护的意图,代码更易于维护和理解。下面举一些实际的例子:
Server结构体中reqLock是用来保护freeReq字段,respLock用来保护freeResp字段,都是将mutex放在被保护字段的上面
  1. //net/rpc server.go
  2. type Server struct {
  3.    serviceMap sync.Map   // map[string]*service
  4.    reqLock    sync.Mutex // protects freeReq
  5.    freeReq    *Request
  6.    respLock   sync.Mutex // protects freeResp
  7.    freeResp   *Response
  8. }
复制代码
在Transport结构体中,idleMu锁会保护closeIdle等一系列字段,此时将锁放在被保护字段的最上面,然后用空格将被idleMu锁保护的字段和其他字段分隔开来。 实现更细粒度的锁定,也能更清晰地表达每个字段需要被互斥保护的意图。
  1. // net/http transport.go
  2. type Transport struct {
  3.    idleMu       sync.Mutex
  4.    closeIdle    bool                                // user has requested to close all idle conns
  5.    idleConn     map[connectMethodKey][]*persistConn // most recently used at end
  6.    idleConnWait map[connectMethodKey]wantConnQueue  // waiting getConns
  7.    idleLRU      connLRU
  8.    reqMu       sync.Mutex
  9.    reqCanceler map[cancelKey]func(error)
  10.    altMu    sync.Mutex   // guards changing altProto only
  11.    altProto atomic.Value // of nil or map[string]RoundTripper, key is URI scheme
  12.    connsPerHostMu   sync.Mutex
  13.    connsPerHost     map[connectMethodKey]int
  14.    connsPerHostWait map[connectMethodKey]wantConnQueue // waiting getConns
  15. }
复制代码
4.3 尽量减小锁的作用范围

在一个代码段里,尽量减小锁的作用范围可以提高并发性能,减少锁的等待时间,从而减少系统资源的浪费。
锁的作用范围越大,那么就有越多的代码需要等待锁,这样就会降低并发性能。因此,在编写代码时,应该尽可能减小锁的作用范围,只在需要保护的临界区内加锁。
如果锁的作用范围是整个函数,使用 defer 语句来释放锁是一种常见的做法,可以避免忘记手动释放锁而导致的死锁等问题。
  1. func (t *Transport) CloseIdleConnections() {
  2.    t.lk.Lock()
  3.    defer t.lk.Unlock()
  4.    if t.idleConn == nil {
  5.       return
  6.    }
  7.    for _, conns := range t.idleConn {
  8.       for _, pconn := range conns {
  9.          pconn.close()
  10.       }
  11.    }
  12.    t.idleConn = nil
  13. }
复制代码
在使用锁时,注意避免在锁内执行长时间运行的代码或者IO操作,因为这样会阻塞锁的使用,导致锁的等待时间变长。如果确实需要在锁内执行长时间运行的代码或者IO操作,可以考虑将锁释放,让其他代码先执行,等待操作完成后再重新获取锁, 比如下面代码示例
  1. // net/http/httputil persist.go
  2. func (cc *ClientConn) Read(req *http.Request) (resp *http.Response, err error) {
  3.    // Retrieve the pipeline ID of this request/response pair
  4.    cc.mu.Lock()
  5.    id, ok := cc.pipereq[req]
  6.    delete(cc.pipereq, req)
  7.    if !ok {
  8.       cc.mu.Unlock()
  9.       return nil, ErrPipeline
  10.    }
  11.    cc.mu.Unlock()
  12.    
  13.     // xxx 省略掉一些中间逻辑
  14.    // 从http连接中读取http响应数据, 这个IO操作,先解锁
  15.    resp, err = http.ReadResponse(r, req)
  16.    // 网络IO操作结束,再继续读取
  17.    
  18.    cc.mu.Lock()
  19.    defer cc.mu.Unlock()
  20.    if err != nil {
  21.       cc.re = err
  22.       return resp, err
  23.    }
  24.    cc.lastbody = resp.Body
  25.    cc.nread++
  26.    if resp.Close {
  27.       cc.re = ErrPersistEOF // don't send any more requests
  28.       return resp, cc.re
  29.    }
  30.    return resp, err
  31. }
复制代码
5.总结

在并发编程中,Mutex是一种常见的同步机制,用来保护共享资源。为了提高并发性能,我们需要尽可能缩小Mutex的锁粒度,只保护需要保护的数据,同时在一个代码段里,尽量减小锁的作用范围。如果锁的作用范围是整个函数,可以使用defer来在函数退出时解锁。当Mutex嵌入到结构体中时,我们可以将Mutex放到要控制的字段上面,并使用空格将字段进行分隔,以便只保护需要保护的数据。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

杀鸡焉用牛刀

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表