Go-知识并发控制RWMutex

embedded/2025/1/16 13:59:00/

Go-知识并发控制RWMutex

  • 1. 介绍
  • 2. 原理
    • 2.1 读写锁的数据结构
    • 2.2 接口定义
    • 2.3 Lock() 写锁定 原理
    • 2.4 Unlock() 写锁定解锁 原理
    • 2.5 RLock() 读锁定 原理
    • 2.6 RUnlock() 读锁定解锁 原理
  • 3. 场景分析
    • 3.1 写锁定如何阻塞写锁定
    • 3.2 写锁定如何阻塞读锁定
    • 3.3 读锁定如何阻塞写锁定
    • 3.4 写锁定不会被 饿死

gitio: https://a18792721831.github.io/

1. 介绍

互斥锁 Mutex 是串行加锁,拿到锁之后,不管是读操作还是写操作,对于 Mutex 来说,是等价的。
但是在并发里面,如果仅仅是读操作,不改变数据的前提下,是可以共享的,多个协程读取到的数据都是可信的。
Mutex 存在这几个问题:

  • 写锁需要阻塞写锁: 一个协程拥有写锁时,其他协程写操作需要阻塞
  • 写锁需要阻塞读锁: 一个协程拥有写锁时,其他协程读操作需要阻塞
  • 读锁需要阻塞写锁: 一个协程拥有读锁时,其他协程写操作需要阻塞
  • 读锁不能阻塞读锁: 一个协程拥有读锁时,其他协程读操作不需要阻塞

RWMutex 是 Mutex 的一个增强版,解决了上面的4个问题

2. 原理

2.1 读写锁的数据结构

在源码包src/sync/rwmutex.go中定义了读写锁的数据结构

type RWMutex struct {w           Mutex  // 用于控制多个写锁,获得写锁需要先获取该锁writerSem   uint32 // 写阻塞等待的信号量,最后一个读者释放锁时会释放信号量readerSem   uint32 // 读阻塞的协程等待的信号量,持有写锁的协程释放锁后会释放信号量readerCount int32  // 记录读者个数readerWait  int32  // 记录写阻塞时读者个数
}

读写锁内部使用 Mutex 互斥锁,用于将多个写操作隔离开来。

2.2 接口定义

RWMutex 提供了这几个接口用于操作锁:

  • RLock(): 读锁定
  • RUnlock(): 读锁定解锁
  • Lock(): 写锁定
  • Unlock(): 解除写锁定
  • TryLock(): 以非阻塞方式尝试写锁定( >= Go 1.18)
  • TryRLock(): 以非阻塞方式尝试读锁定( >= Go 1.18)

2.3 Lock() 写锁定 原理

写锁定需要实现两个动作: 1. 获取互斥锁;2. 阻塞等待所有读操作结束(如果存在未结束的读操作)

// 锁定rw进行写入。
// 如果锁已经被锁定用于读取或写入,
// 锁定块,直到锁定可用。
func (rw *RWMutex) Lock() {// 首先,解决与其他作家的竞争。// rw.w 是 Mutex 锁,Lock 操作 会阻塞获取锁// 如果有多个写操作竞争锁,那么会使用 Mutex 的策略,进行排队或自旋 获取锁rw.w.Lock()// 向读者宣布有一个待定的作者。// 假设 raderCount = 2 , rwmutexMaxReaders = 100// 并发安全的 2 - 100 = -98 // 并发不安全的 -98 + 100 = 2 // 拿到了 raderCount// 因为 raderCount 是 int32 类型,并不是 atomic.Int32,所以不能使用 Load() 方法并发安全的获取值// func AddInt32(addr *int32, delta int32) (new int32) // 直接调用底层方法,让一个普通的 int32 拥有了 atomic 的能力r := atomic.AddInt32(&rw.readerCount, -rwmutexMaxReaders) + rwmutexMaxReaders//等待活动的读卡器。// 如果读操作不为空,那么将当前写操作记录到 写阻塞时,读操作的个数if r != 0 && atomic.AddInt32(&rw.readerWait, r) != 0 {// 等待读操作结束后释放信号量,这里是阻塞的,等待 writerSem 地址的信号量runtime_SemacquireMutex(&rw.writerSem, false, 0)}
}

需要注意一点,在 Lock() 操作里面 readerCount 因为溢出,现在是负值

最开始的时候, readerCount 是 0 , 当 Lock 后,是 负的 max
在这里插入图片描述

2.4 Unlock() 写锁定解锁 原理

写锁定解锁要实现的两个动作: 1. 唤醒因读锁定而被阻塞的协程(如果有)(写操作的同时发生的读操作,因为写操作还未完成,所以读操作需要等待写操作完成后才能进行读); 2. 解除互斥锁(写操作必须先获取 Mutex )

//解锁解锁rw进行写入。如果rw为
//未锁定,无法在输入解锁时写入。
//
//与Mutex一样,锁定的RWMutex与特定的
//goroutine。一个goroutine可以RLock(Lock)一个RWMutex,然后
//安排另一个goroutine来解锁它。
func (rw *RWMutex) Unlock() {//向读者宣布没有活动的写入程序。// 因为在 Lock 中, raderCount 的值被置为负值// 所以在这里在 加上 rwmutexMaxReaders ,readerCount 的值就变成了正值r := atomic.AddInt32(&rw.readerCount, rwmutexMaxReaders)// 如果出现溢出,,那么是错误的if r >= rwmutexMaxReaders {throw("sync: Unlock of unlocked RWMutex")}// 取消阻止被阻止的读卡器(如果有的话)。// 因为存在 r 个读操作因为写操作而阻塞,当写操作完成时,就需要唤醒读操作for i := 0; i < int(r); i++ {runtime_Semrelease(&rw.readerSem, false, 0)}//允许其他写入程序继续。// 释放锁,等待下次写操作rw.w.Unlock()
}

这里其实有一点小的不同,当写操作锁释放时,如果即存在读操作等待,又存在写操作等待,那么是优先读操作的。
也就是: 读 -> 写 -> 读 -> …

这里引申出来一个思考,假设在读操作中,不断的派生,导致读操作占用一直无法结束,那么就会导致写操作加锁也无法加锁
写锁定解除的时候,将 readerCount + max ,获取到真正的 reader waiter
然后释放 reader waiter 个信号量,唤醒全部等待的读锁定
在这里插入图片描述

2.5 RLock() 读锁定 原理

读操作加锁需要的动作: 1. 增加读操作计数器; 2. 阻塞等待写锁定解锁(如果有)

// 在关系通过以下方式指示给竞赛检测器之前发生:
// - Unlock  -> Lock:  readerSem
// - Unlock  -> RLock: readerSem
// - RUnlock -> Lock:  writerSem
//
// 以下方法暂时禁用竞赛同步的处理
// 事件,以便为比赛提供更精确的模型 探测器
//
// 例如,原子。RLock中的AddInt32不应显示为提供
// 获取发布语义,这将错误地同步比赛
// 读者,从而可能错过种族。
// RLock锁定rw进行读取。
//
// 它不应用于递归读取锁定;被锁住的锁
// 调用将阻止新读者获取锁。请参阅
// RWMutex类型的文档。
func (rw *RWMutex) RLock() {// 在写锁定里面,会将 raderCount 置为负数,所以如果给 raderCount + 1 依然小于 0 // 那么说明至少有一个读锁定在阻塞等待写锁定解锁// 换句话说,当 raderCount + 1 小于 0 ,说明有协程正在 写锁定if atomic.AddInt32(&rw.readerCount, 1) < 0 {//有一个写入程序正在挂起,请稍候。runtime_SemacquireMutex(&rw.readerSem, false, 0)}
}

因为在 写锁定 操作中,是直接减去最大值,所以这里实际上是不会限制读锁加锁数量的,也就是说,如果不断派生,那么读操作可能永远无法结束。
写锁定饿死?
这里要分为两种场景:

  1. RWMutex 无锁定,那么 readerCount > 0 ,就是读锁定个数
  2. RWMutex 有写锁定,那么 readerCount < 0 , 读锁定个数 = readerCount + max
    在这里插入图片描述

2.6 RUnlock() 读锁定解锁 原理

读锁定解锁的动作: 1. 减少读操作计数器; 2. 唤醒等待的写锁定(如果有)

// RUnlock撤消单个RLock调用;
// 它不会影响其他同时阅读的人。
// 如果rw未锁定以进行读取,则是运行时错误
// 在进入RUnlock时。
func (rw *RWMutex) RUnlock() {// 在 RUnlock 中,readerCount 是不能变为负值的,如果变为负值,// 说明读锁在未拿到锁的情况下,调用了 RUnlock// 也有可能是还未加锁就解锁,加锁 readerCount++, 解锁 readerCount--// 还有一种情况是,存在读锁定的时候,出现了写锁定,写锁定会将 readerCount 变为负值if r := atomic.AddInt32(&rw.readerCount, -1); r < 0 {//勾勒出慢速路径以允许快速路径内联rw.rUnlockSlow(r)}
}
func (rw *RWMutex) rUnlockSlow(r int32) {// 如果 r - 1 = -1 那么说明是未加锁就解锁,异常// 如果 r + 1 = - max , 加锁数量超过最大值,异常if r+1 == 0 || r+1 == -rwmutexMaxReaders {throw("sync: RUnlock of unlocked RWMutex")}// A writer is pending.// 如果 是 写锁定情况下 读锁定解锁,此时会判断 readerWait 的值// 防止 写锁定 饿死,在写锁定的时候,会同步 readerWait = readerCount// 比如 写锁定时 readerCount = n , 此时 readerWait = n ,同时会将 readerCount 变为负值// 变为负值后,读锁定就会阻塞// 然后等待 读锁定解锁// 在读锁定解锁的时候,进入 rUnlockSlow 就说明,有 写锁定 在阻塞// 此时 readerWait > 0 , 那么每次读锁定解锁,会将 readerWait--// 当 readerWait 等于 0  的时候,需要唤醒写锁定// 用这样的逻辑防止 写锁定 饿死// 只有最后一个才唤醒if atomic.AddInt32(&rw.readerWait, -1) == 0 {// The last reader unblocks the writer.runtime_Semrelease(&rw.writerSem, false, 1)}
}

在这里插入图片描述

3. 场景分析

3.1 写锁定如何阻塞写锁定

RWMutex 包含了一个 Mutex ,获取写锁定必须获取 Mutex ,如果已经有写锁定获取了 Mutex
那么后续的写锁定就必须等待前面的写锁定释放 Mutex 才能继续写锁定。

3.2 写锁定如何阻塞读锁定

写锁定会将 readerCount 的值变为负值,在读锁定里面,如果 readerCount + 1 < 0 那么会进入阻塞。
将 readerCount 的值变为负值,不会导致 readerCount 的原始值丢失,只需要 readerCount + max
就能重新拿到真实的值,并且期间针对 readerCount 的运算结果也能继续保留下来。

这个实现非常精妙,值得学习。

3.3 读锁定如何阻塞写锁定

读锁定会将 readerCount + 1 ,写锁定读取到 readerCount 不为 0 ,就会阻塞,
同时设置 readerWait = readerCount ,防止写锁定饿死

3.4 写锁定不会被 饿死

写锁定需要等待读锁定解锁后才能获取锁,写锁定的等待期间可能还有新的读锁定,那么可能永远等不到全部的读锁定解锁。
这样写锁定就出现 饿死
但是实际上是写锁定会将 readerCount 的值放到 readerWait 中,并且将 readerCount 的值变为负值
当 readerCount 的值变为负值,就不会有新的 读锁定了,读锁定因为 readerCount + 1 < 0 而进入阻塞
在 读锁定解锁的时候,如果 readerWait + 1 = 0 ,就会唤醒写锁定
相当于 当读锁定还未解锁时,写锁定会将当前读锁定的数量记下来,然后阻塞新增新的读锁定
等待记录的读锁定解锁后,就进行写锁定
强行将读锁定分割,避免饿死


http://www.ppmy.cn/embedded/48862.html

相关文章

苹果WWDC 2024 带来的 AI 风暴:从生产力工具到个人助理,AI 将如何融入我们的生活?

2024年6月5日&#xff0c;苹果WWDC 2024全球开发者大会如约而至&#xff0c;带来了众多令人兴奋的新功能和新产品。其中&#xff0c;AI 技术的全面融入无疑是最引人注目的亮点。从 iOS、iPadOS 到 macOS&#xff0c;再到 Siri 和开发者工具&#xff0c;苹果正在将 AI 融入到其生…

ubuntu 22.04下利用webmin 搭建一个Wordpress 网站(2)

上次我们讲到第二部分&#xff0c;今天我们继续这一个话题 第三部分&#xff1a;利用webmin创建一个wordpress网站 1、在 Webmin 内安裝Apache 未使用的模块> Apache Webserver > 现在安装 会出现如下图所示的有关软件 刷新模快后 检查开机时要自动启动Apache 测…

【线性代数】向量空间,子空间,向量空间的基和维数

向量空间 设V为n维向量的集合&#xff0c;如果V非空&#xff0c;且集合V对于向量的加法以及数乘两种运算封闭&#xff0c;那么就称集合V为向量空间 x&#xff0c;y是n维列向量。 x 向量组等价说明可以互相线性表示 向量组等价则生成的向量空间是一样的 子空间 例题18是三位向…

接口自动化Requests+Pytest基础实现

目录 1. 数据库以及数据库操作1.1 概念1.2 分类1.3 作用 2 python操作数据库的相关实现2.1 背景2.2 相关实现 3. pymysql基础3.1 整个流程3.2 案例3.3 Pymysql工具类封装 4 事务4.1 案例4.2 事务概念4.3 事务特征 5. requests库5.1 概念5.2 角色定位5.3 安装5.4 校验5.5 reques…

微信小程序毕业设计-实验室管理系统项目开发实战(附源码+论文)

大家好&#xff01;我是程序猿老A&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f49e;当前专栏&#xff1a;微信小程序毕业设计 精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; &#x1f380; Python毕业设计…

ARP协议相关

把ip地址解析成mac地址这里的mac地址就是路由器的mac地址 免费ARP 源ip和目的ip都是一样的&#xff0c;那怎么让其他人更新arp表呢&#xff1f;&#xff1f; 是因为目标mac是全f&#xff0c;是一个广播报文 如果冲突就是ip一样但是mac又不一样 代理ARP pc1和pc4是在同一个子网…

第一页总结

第一页总结 链表反转206. 反转链表25. K 个一组翻转链表 双指针21. 合并两个有序链表141. 环形链表 二叉树102. 二叉树的层序遍历236. 二叉树的最近公共祖先 数组1.两数之和15. 三数之和 链表 反转 206. 反转链表 206. 反转链表 给你单链表的头节点 head &#xff0c;请你反…

人类如何挣脱被人工智能替代的命运?

人工智能技术的迭代升级&#xff0c;使得“换脸”“拟声”成为可能&#xff0c;我如何证明不是“我”&#xff1f;面对人工智能超高的生产效率&#xff0c;我如何与人工智能“抢工作”&#xff1f;在人工智能时代&#xff0c;如何回应这类疑问&#xff1f;挣脱被替代的命运&…