Mutex用于互斥访问,RWMutex提升读多写少场景性能;使用defer确保解锁,避免死锁与复制Mutex,推荐-race检测竞争。
在go语言中,sync 包提供了多种同步原语,用于协调多个goroutine之间的并发访问。其中最常用的两种是 Mutex 和 RWMutex,它们用于保护共享资源,防止数据竞争。
Mutex:互斥锁
Mutex(Mutual Exclusion)是互斥锁,用于确保同一时间只有一个goroutine可以访问临界区。
典型使用方式是在访问共享变量前加锁,操作完成后立即解锁:
var mu sync.Mutex
立即学习“go语言免费学习笔记(深入)”;
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
// 读操作
rwmu.RLock()
defer rwmu.RUnlock()
value := config[“key”]
// 写操作
rwmu.Lock()
defer rwmu.Unlock()
config[“key”] = “new_value”
写锁是排他性的,只要有写锁存在,其他读和写都会被阻塞。读锁之间不互斥,适合高并发读场景。
使用建议与注意事项
避免锁粒度过大,只保护真正需要同步的代码段,减少性能损耗。
不要在持有锁期间做耗时操作,比如网络请求或长时间计算。
避免死锁:多个锁要按固定顺序加锁;不要在锁内调用可能加锁的函数。
复制包含Mutex的结构体可能导致程序异常,因为Mutex不支持拷贝。应始终通过指针传递。
调试时可使用 -race 参数运行程序,检测数据竞争问题。
基本上就这些。Mutex适合简单互斥场景,RWMutex在读多写少时更高效。合理使用,能有效保障并发安全。