Golang的sync同步原语 Mutex/RWMutex使用

Mutex用于互斥访问,RWMutex提升读多写少场景性能;使用defer确保解锁,避免死锁与复制Mutex,推荐-race检测竞争。

Golang的sync同步原语 Mutex/RWMutex使用

go语言中,sync 包提供了多种同步原语,用于协调多个goroutine之间的并发访问。其中最常用的两种是 MutexRWMutex,它们用于保护共享资源,防止数据竞争。

Mutex:互斥锁

Mutex(Mutual Exclusion)是互斥锁,用于确保同一时间只有一个goroutine可以访问临界区。

典型使用方式是在访问共享变量前加锁,操作完成后立即解锁:

var mu sync.Mutex

立即学习go语言免费学习笔记(深入)”;

var count int

mu.Lock()

count++

mu.Unlock()

注意:务必保证 Unlock() 能被执行,通常使用 defer 来避免死锁:

mu.Lock()

defer mu.Unlock()

count++

如果忘记解锁,其他goroutine将永久阻塞,导致程序卡死。重复解锁会引发panic。

RWMutex:读写锁

当共享资源的读操作远多于写操作时,使用 RWMutex 可以提升并发性能。

RWMutex 区分两种锁:

  • 读锁(RLock/RLocker):多个goroutine可同时持有读锁
  • 写锁(Lock):写操作独占,阻塞其他读和写

适用场景:缓存、配置中心、频繁读取的状态变量。

var rwmu sync.RWMutex

var config map[String]string

// 读操作

rwmu.RLock()

defer rwmu.RUnlock()

value := config[“key”]

// 写操作

rwmu.Lock()

defer rwmu.Unlock()

config[“key”] = “new_value”

写锁是排他性的,只要有写锁存在,其他读和写都会被阻塞。读锁之间不互斥,适合高并发读场景。

使用建议与注意事项

避免锁粒度过大,只保护真正需要同步的代码段,减少性能损耗。

不要在持有锁期间做耗时操作,比如网络请求或长时间计算。

避免死锁:多个锁要按固定顺序加锁;不要在锁内调用可能加锁的函数。

复制包含Mutex的结构体可能导致程序异常,因为Mutex不支持拷贝。应始终通过指针传递。

调试时可使用 -race 参数运行程序,检测数据竞争问题。

基本上就这些。Mutex适合简单互斥场景,RWMutex在读多写少时更高效。合理使用,能有效保障并发安全。

© 版权声明
THE END
喜欢就支持一下吧
点赞14 分享