RWMutex适用于读多写少场景,允许多读单写以提升并发性能;需避免写饥饿、禁止读锁中写操作或嵌套锁,零值有效且无需初始化。

Go 语言中,RWMutex(读写互斥锁)是 sync 包提供的核心并发原语之一,专为“读多写少”场景优化。它允许多个 goroutine 同时读,但写操作必须独占,从而在保障数据安全的前提下显著提升读操作的并发性能。
什么时候该用 RWMutex 而不是 Mutex
当你有以下特征时,RWMutex 更合适:
- 共享数据被读取的频率远高于被修改的频率(比如配置缓存、状态快照、只读映射表)
- 读操作耗时较长或并发量大,用普通
Mutex会导致读之间互相阻塞 - 写操作不频繁,且每次写入逻辑可控(避免写饥饿,后文会说明)
注意:RWMutex 并不比 Mutex “更快”,它只是把读/写路径做了分离;如果读写一样频繁,甚至写更多,它反而可能因额外开销而更慢。
RWMutex 的基本用法和关键方法
sync.RWMutex 提供 5 个核心方法,使用时需严格配对:
立即学习“go语言免费学习笔记(深入)”;
-
RLock()和RUnlock():获取/释放读锁(可重入,多个 goroutine 可同时持有) -
Lock()和Unlock():获取/释放写锁(排他,任一时刻最多一个) -
RLocker():返回一个Locker接口,方便传给期望sync.Locker的函数(如defer mu.RLocker().Unlock()不推荐,易出错,建议显式调用)
典型模式:
var mu sync.RWMutex
var data map[string]int
// 读操作
func Get(key string) (int, bool) {
mu.RLock()
defer mu.RUnlock()
v, ok := data[key]
return v, ok
}
// 写操作
func Set(key string, val int) {
mu.Lock()
defer mu.Unlock()
data[key] = val
}
写操作会阻塞新读锁?理解锁的公平性
RWMutex 默认**不保证完全公平**。它的行为是:
- 多个读锁可以并行,只要没写锁在等待
- 一旦有 goroutine 调用了
Lock(),后续的RLock()会被阻塞,直到该写锁完成 - 但已持有的读锁不会被强制释放——也就是说,写锁需等待所有当前活跃读锁全部
RUnlock()后才能进入
这意味着:如果读操作持续不断(比如高频轮询),写操作可能长时间等待,即“写饥饿”。若业务不能容忍,可考虑:
- 限制读操作频率或生命周期(避免长时持锁)
- 改用带超时的控制逻辑(如结合
context+ 定时重试) - 评估是否真需要强一致性,有时用原子值(
atomic.Value)做无锁快照更合适
常见陷阱与建议
几个容易踩的坑:
- 不要在持有读锁时调用可能阻塞或耗时的操作(如 HTTP 请求、数据库查询、channel receive),否则会拖慢其他读协程
-
禁止嵌套锁:比如在
RLock()之后再调用Lock(),会死锁(Go 运行时不会报错,但会永久阻塞) - 写锁期间可安全读写,但读锁期间只应读,不可写——虽然语法允许,但违反约定会破坏线程安全
-
零值
RWMutex是有效的,无需显式初始化(var mu sync.RWMutex即可)
进阶提示:若需更细粒度控制(如分片读写锁、带版本的乐观读),可组合 atomic 或考虑第三方库如 github.com/jonhoo/fnmatch 类型的专用结构,但大多数场景原生 RWMutex 已足够。
基本上就这些。RWMutex 不复杂,但容易忽略其调度特性和使用边界。用对了,能让你的并发代码既安全又高效。










