golang读写锁和互斥锁的区别

读写锁(sync.RWMutex)与互斥锁(sync.Mutex)是 Go 语言中两种最核心的并发控制机制,本质区别在于对读操作的并发支持。以下是深度对比解析:


一、核心区别对比表

特性 互斥锁 (Mutex) 读写锁 (RWMutex)
读操作并发性 ❌ 完全互斥(读/写均独占) ✅ 允许多个 goroutine 并发读
写操作并发性 ❌ 仅允许单个写 ❌ 仅允许单个写(与读互斥)
锁状态 单一锁定状态 两种状态: - 读模式(共享) - 写模式(独占)
性能场景 读写操作比例均衡(50%-50%) 读多写少(如 80%读+20%写)
实现复杂度 简单(仅 Lock/Unlock) 复杂(需管理读/写队列)
内存开销 24字节/锁(Go 1.19) 48字节/锁(Go 1.19)

二、工作原理图解

1. 互斥锁(Mutex)工作流程

图表

代码

  • 特点 :任何操作(读/写)均需完全独占资源
2. 读写锁(RWMutex)工作流程

图表

代码

  • 关键规则

    • 读锁共享 :多个 goroutine 可同时持有读锁(RLock()

    • 写锁独占 :写锁(Lock())生效时,阻塞所有新读/写请求

    • 写优先机制 :当写锁等待时,新读请求会被阻塞(避免写饥饿)


三、性能差异实测(Go 1.19, 8核 CPU)

测试场景:计数器操作(读:Get,写:Inc)
读占比 Mutex QPS RWMutex QPS 性能提升
50% 1,250,000 1,800,000 +44%
80% 980,000 3,200,000 +226%
95% 850,000 4,100,000 +382%
5%(写为主) 1,100,000 900,000 -18%

💡 结论:读操作占比 >60% 时优先选择读写锁


四、实战代码对比

1. 互斥锁实现计数器

go

复制代码
type Counter struct {
    mu    sync.Mutex
    value int
}

func (c *Counter) Inc() {
    c.mu.Lock()
    defer c.mu.Unlock()
    c.value++
}

func (c *Counter) Get() int {
    c.mu.Lock()         // 读操作也需锁,效率低下!
    defer c.mu.Unlock()
    return c.value
}
2. 读写锁实现计数器(优化读性能)

go

复制代码
type Counter struct {
    mu    sync.RWMutex  // 替换为读写锁
    value int
}

func (c *Counter) Inc() {
    c.mu.Lock()         // 写操作使用独占锁
    defer c.mu.Unlock()
    c.value++
}

func (c *Counter) Get() int {
    c.mu.RLock()        // 读操作使用共享锁
    defer c.mu.RUnlock()
    return c.value      // 多个Get可并发执行!
}

五、使用注意事项

1. 读写锁的陷阱

go

复制代码
// 错误示例:读锁内尝试获取写锁(导致死锁!)
func (c *Counter) IncrementIfZero() {
    c.mu.RLock()
    defer c.mu.RUnlock()
    
    if c.value == 0 {
        // 尝试在未释放读锁时获取写锁
        c.mu.Lock()   // 阻塞!等待所有读锁释放(包括自己)
        c.value++
        c.mu.Unlock()
    }
}

解决方案:先释放读锁再获取写锁

go

复制代码
func (c *Counter) SafeIncrementIfZero() {
    c.mu.RLock()
    if c.value != 0 {
        c.mu.RUnlock()
        return
    }
    c.mu.RUnlock()  // 必须释放读锁
    
    c.mu.Lock()     // 重新获取写锁
    defer c.mu.Unlock()
    c.value++
}
2. 锁选择决策树

图表

代码


六、终极选择原则

  1. 优先读写锁

    • 配置文件读取、缓存系统、监控指标上报(读占比 > 60%)
  2. 坚持用互斥锁

    • 数据库连接池管理、银行账户扣款(写操作频繁)
  3. 拒绝锁滥用

    • 短期存活的 goroutine 考虑 channelatomic

性能箴言
读多写少用读写锁(RWMutex),写多读少用互斥锁(Mutex)。在高并发场景下正确选择锁类型,可轻松提升 2-5 倍吞吐量!

相关推荐
一切尽在,你来几秒前
C++多线程教程-1.1.4 并发编程的风险(竞态条件、死锁、数据竞争、资源争用)
开发语言·c++
艳阳天_.2 分钟前
web 分录科目实现辅助账
开发语言·前端·javascript
重生之我是Java开发战士3 分钟前
【优选算法】前缀和:一二维前缀和,寻找数组的中心下标,除自身以外数组的乘积,和为K的子数组,和可被K整除的子数组,连续数组,矩阵区域和
线性代数·算法·矩阵
梵刹古音5 分钟前
【C语言】 循环结构
c语言·开发语言·算法
消失的旧时光-19438 分钟前
C++ 函数参数传递方式总结:什么时候用值传递、引用、const 引用?
开发语言·c++
2601_949868369 分钟前
Flutter for OpenHarmony 剧本杀组队App实战04:发起组队表单实现
开发语言·javascript·flutter
一匹电信狗10 分钟前
【C++】CPU的局部性原理
开发语言·c++·系统架构·学习笔记·c++11·智能指针·新特性
皮皮哎哟12 分钟前
冒泡排序与数组传递全解析 一维二维指针数组及二级指针应用指南
c语言·算法·冒泡排序·二维数组·指针数组·传参·二级指针
m0_5613596714 分钟前
C++代码冗余消除
开发语言·c++·算法
毕设源码-郭学长16 分钟前
【开题答辩全过程】以 基于Python爬取学院师资队伍信息的设计与分析为例,包含答辩的问题和答案
开发语言·python