Go 语言中 sync.RWMutex 实现读写锁,允许多读并发、写独占,适用于读多写少场景;需正确配对 RLock/RUnlock 和 Lock/Unlock,避免混用、嵌套或读锁内耗时操作。

Go 语言中通过 sync.RWMutex 实现读写锁,让多个读操作可以并行执行,而写操作独占资源,从而显著提升高读低写的并发场景效率。
理解读写锁的核心逻辑
读写锁不是“升级”或“降级”互斥锁,而是将访问分为两类:读操作(共享) 和 写操作(独占)。只要没有写操作进行,任意数量的 goroutine 可以同时读;一旦有写操作开始,所有新读、新写都必须等待;写操作完成前,不允许其他任何读或写进入。
注意:sync.RWMutex 不保证公平性,可能造成写饥饿(大量读持续到来时,写一直被阻塞),如需公平控制,需额外加队列或使用第三方库(如 github.com/cespare/xxhash 配合自定义调度)。
基本用法:正确调用 RLock / RUnlock 和 Lock / Unlock
读操作使用 RLock() 和 RUnlock(),写操作使用 Lock() 和 Unlock()。二者不可混用,且必须成对出现,否则会导致 panic 或死锁。
立即学习“go语言免费学习笔记(深入)”;
- 读操作示例:
defer mu.RUnlock()
// 安全读取共享数据,如 map、slice 等
- 写操作示例:
defer mu.Unlock()
// 修改共享数据,如赋值、删除、扩容等
切勿在持有 RLock 期间调用 Lock,这会死锁;也不要在 Lock 持有时再 RLock——RWMutex 不支持递归或嵌套。
典型适用场景与避坑提醒
适合读多写少、数据结构生命周期长、读操作耗时短的场景,例如配置缓存、路由表、状态快照等。
- 避免在读锁内做耗时操作(如 HTTP 请求、数据库查询),否则会阻塞其他写操作和后续读操作
- 不要用 RWMutex 保护需要原子更新的字段(如计数器),应改用
sync/atomic - map 类型不能直接用 RWMutex 保护其并发读写——必须确保 map 本身不扩容(即初始化后容量固定),否则仍可能 panic;更安全的做法是用
sync.Map或封装为只读快照 + 写时重建
一个简洁实用的配置管理示例
以下是一个带读写锁的线程安全配置结构:
type Config struct {mu sync.RWMutex
data map[string]string
}
func (c *Config) Get(key string) string {
c.mu.RLock()
defer c.mu.RUnlock()
return c.data[key]
}
func (c *Config) Set(key, value string) {
c.mu.Lock()
defer c.mu.Unlock()
c.data[key] = value
}
若配置更新频率极低,还可进一步优化:写时生成新 map,读时原子替换指针(配合 atomic.Value),彻底消除读锁开销。










