RWMutex —— 细粒度的读写锁

我们之前有讲过 Mutex 互斥锁。这是在任何时刻下只允许一个 goroutine 执行的串行化的锁。而现在这个 RWMutex 就是在 Mutex 的基础上进行了拓展能支持多个 goroutine 持有读锁,而在尝试持有写锁时就会如 Mutex 一样就会陷入等待锁的释放。它是一种细粒度的锁。虽然可以允许多次持有读锁,但是 Go 团队还特意嘱咐,为了确保锁的可用性,不能用于递归读锁。一个阻塞的锁要排除正在持有锁的新读。

那么上面说到的这些功能,RWMutex 是如何实现的呢?首先我们来看它的内部结构:

type RWMutex struct {
w Mutex // held if there are pending writers
writerSem uint32 // semaphore for writers to wait for completing readers
readerSem uint32 // semaphore for readers to wait for completing writers
readerCount int32 // number of pending readers
readerWait int32 // number of departing readers
}

只有 5 个对象,其中最重要的就是 Mutex 锁的字段 w,它就是实现写锁的关键。

  • writerSem 是写等待读完成的信号量
  • readerSem 是读等待写完成的信号量
  • readerCount 正处于读锁的个数
  • readerWait 尝试获取写锁时读等待的个数(这个怎么理解?)

其中还有一个全局的常数变量 rwmutexMaxReaders,表示最多的读操作。

我们先来看写锁

Lock/UnLock 写锁/解锁

func (rw *RWMutex) Lock() {
...
rw.w.Lock()
r := atomic.AddInt32(&rw.readerCount, -rwmutexMaxReaders) + rwmutexMaxReaders
// Wait for active readers.
if r != 0 && atomic.AddInt32(&rw.readerWait, r) != 0 {
runtime_SemacquireMutex(&rw.writerSem, false, 0)
}
if race.Enabled {
race.Enable()
race.Acquire(unsafe.Pointer(&rw.readerSem))
race.Acquire(unsafe.Pointer(&rw.writerSem))
}
}

这里直接用到了 Mutex 互斥锁来保证只有一个 goroutine 能进来。接下来就会判断在获取写锁的时候如果还存在其他的读锁没有释放,那么这个时候就会陷入睡眠进入等待者队列中等待所有的读锁被释放之后唤醒

可能有些人对这个限制有些不懂,其实这就是为了保证锁的区间的读的值顺序性的正确性。因为在获取写的时候,目的就是进行写操作,所谓我就必须要在此时还存在其他可能会读这个变量的读锁全部释放才行。

而释放写锁就是 UnLock 操作了。如果调用此操作时,本就没有上锁那么就会直接抛异常。

func (rw *RWMutex) Unlock() {
...
// Announce to readers there is no active writer.
r := atomic.AddInt32(&rw.readerCount, rwmutexMaxReaders)
if r >= rwmutexMaxReaders {
race.Enable()
throw("sync: Unlock of unlocked RWMutex")
}
// Unblock blocked readers, if any.
for i := 0; i < int(r); i++ {
runtime_Semrelease(&rw.readerSem, false, 0)
}
// Allow other writers to proceed.
rw.w.Unlock()
...
}

如果还存在读锁时,那么就会进入 runtime.Semrelease 对那些阻塞的读锁解锁(找到对应的信号量等待者队列然后弹出唤醒)。最后释放 w 锁。

RLock/RUnlock 读锁/解锁

func (rw *RWMutex) RLock() {
...
if atomic.AddInt32(&rw.readerCount, 1) < 0 {
// A writer is pending, wait for it.
runtime_SemacquireMutex(&rw.readerSem, false, 0)
}
...
}

读锁就非常简单了,仅仅只是对 readerCount 字段自增。这里的判断要注意,这个判断成立说明有协程调用了 rw.Lock 获取了写锁。所以就要等待其它协程的释放。

知道读锁的机制,那么就能想到释放读锁其实就是撤销读锁,将 readerCount 字段减1即可。

func (rw *RWMutex) RUnlock() {
...
if r := atomic.AddInt32(&rw.readerCount, -1); r < 0 {
// Outlined slow-path to allow the fast-path to be inlined
rw.rUnlockSlow(r)
}
...
}

同样在释放读锁时会判断 r 是否为负数,如果为负数就说明有其它协程获取了写锁,就会进入 rUnlockSlow 方法。

func (rw *RWMutex) rUnlockSlow(r int32) {
if r+1 == 0 || r+1 == -rwmutexMaxReaders {
race.Enable()
throw("sync: RUnlock of unlocked RWMutex")
}
// A writer is pending.
if atomic.AddInt32(&rw.readerWait, -1) == 0 {
// The last reader unblocks the writer.
runtime_Semrelease(&rw.writerSem, false, 1)
}
}

如果锁状态已经是解锁状态则抛异常。

如果是只剩下一个读等待,则释放写信号量通知其他正在尝试持有写锁的协程上锁。

关于信号量的细节

我们上面分析了读写锁的上锁与解锁的过程,其实有一个点不知道大家有没有注意。就是关于信号量的操作对象的细节。

  1. 调用 Lock 获取写锁,会持有 writerSem 信号
  2. 调用 Unlock 释放写锁时,会释放 readerSem 信号
  3. 调用 RLock 获取读锁时,会持有 readerSem 信号
  4. 调用 RUnlock 释放读锁时,会释放 writerSem 信号

大家有没有发现其中的规律,这么做的目的是什么呢?

也就是说:我们在获取写锁之前,会先等待读锁的释放操作。而在获取读锁时,会先等待写锁的释放操作。

我们用反证法来假设这个场景:我这里有一个连续的写操作;那么也就是说我要连续反复的调用 Lock + Unlock 操作。如果没有上面的信号量的互相牵制,那么就很容易出现读操作没法执行的问题,也就是说会”饿死“。

所以 RWMutex 加入读写信号量的机制是为了更好达到 RW 的目的,而不是一直 W。

总结

  • 在调用 Lock 获取写锁时,会先等待 RUnlock 将其 readerCount 置为 0,然后成功获取写锁。

    • 还有一个操作是将 readerCount - rwmutexMaxReaders,其目的是为了阻塞后续的 RLock 操作。即在读取写锁其他任何读写操作都不允许了。
  • 在调用 Unlock 释放写锁时,会通知所有读操作,解锁那些阻塞的读锁,然后成功释放写锁。

RWLock——一种细粒度的Mutex互斥锁的更多相关文章

  1. golang mutex互斥锁分析

    互斥锁:没有读锁写锁之分,同一时刻,只能有一个gorutine获取一把锁 数据结构设计: type Mutex struct { state int32 // 将一个32位整数拆分为 当前阻塞的gor ...

  2. Go 标准库 —— sync.Mutex 互斥锁

    Mutex 是一个互斥锁,可以创建为其他结构体的字段:零值为解锁状态.Mutex 类型的锁和线程无关,可以由不同的线程加锁和解锁. 方法 func (*Mutex) Lock func (m *Mut ...

  3. C# Mutex互斥锁

    Mutex 构造函数 (Boolean, String, Boolean) public Mutex ( bool initiallyOwned, string name, out bool crea ...

  4. C# mutex互斥锁构造

    概念 Mutext 出现的比monitor更早,而且传承自COM,当然,waitHandle也是它的父类,它继承了其父类的功能,有趣的是Mutex的脾气非常的古怪,它 允许同一个线程多次重复访问共享区 ...

  5. 【转】【C#】【Thread】Mutex 互斥锁

    Mutex:互斥(体) 又称同步基元. 当创建一个应用程序类时,将同时创建一个系统范围内的命名的Mutex对象.这个互斥元在整个操作系统中都是可见的.当已经存在一个同名的互斥元时,构造函数将会输出一个 ...

  6. linux c 线程间同步(通信)的几种方法--互斥锁,条件变量,信号量,读写锁

    Linux下提供了多种方式来处理线程同步,最常用的是互斥锁.条件变量.信号量和读写锁. 下面是思维导图:  一.互斥锁(mutex)  锁机制是同一时刻只允许一个线程执行一个关键部分的代码. 1 . ...

  7. Golang 读写锁RWMutex 互斥锁Mutex 源码详解

    前言 Golang中有两种类型的锁,Mutex (互斥锁)和RWMutex(读写锁)对于这两种锁的使用这里就不多说了,本文主要侧重于从源码的角度分析这两种锁的具体实现. 引子问题 我一般喜欢带着问题去 ...

  8. 探索互斥锁 Mutex 实现原理

    Mutex 互斥锁 概要描述 mutex 是 go 提供的同步原语.用于多个协程之间的同步协作.在大多数底层框架代码中都会用到这个锁. mutex 总过有三个状态 mutexLocked: 表示占有锁 ...

  9. 互斥锁Mutex与信号量Semaphore的区别

    转自互斥锁Mutex与信号量Semaphore的区别 多线程编程中,常常会遇到这两个概念:Mutex和Semaphore,两者之间区别如下: 有人做过如下类比: Mutex是一把钥匙,一个人拿了就可进 ...

随机推荐

  1. zabbix企业级的分布式开源监控解决方案 v5.0 LTS

    目录 zabbix简介 服务模块 客户端守护进程 监控流程 功能拆解 安装 zabbix 5.0 LTS 参考官网 zabbix 5.0.12-1.el7 zabbix-server相关优化 1. 字 ...

  2. [转]CAP和BASE理论

    1. CAP理论 2000年7月,加州大学伯克利分校的Eric Brewer教授在ACM PODC会议上提出CAP猜想.2年后,麻省理工学院的Seth Gilbert和Nancy Lynch从理论上证 ...

  3. Go语言的GOPATH详解

    在GOLAND中设置GOPATH: 设置好路径后,并不是直接在这个路径下面写代码文件就行了 GO会识别GOPATH下的src目录,而真正的引用的包名,是src下的目录名,然后才是代码模块名 目录结构如 ...

  4. 构建可扩展的GPU加速应用程序(NVIDIA HPC)

    构建可扩展的GPU加速应用程序(NVIDIA HPC) 研究人员.科学家和开发人员正在通过加速NVIDIA GPU上的高性能计算(HPC)应用来推进科学发展,NVIDIA GPU具有处理当今最具挑战性 ...

  5. Spring Cloud06: Ribbon 负载均衡

    一.使用背景 前面的学习中,我们已经使用RestTemplate来实现了服务消费者对服务提供者的调用,如果在某个具体的业务场景下,对某个服务的调用量突然大幅提升,这个时候就需要对该服务实现负载均衡以满 ...

  6. 在H5页面播放m3u8音频文件

    需要使用hls插件 首先安装依赖npm install hls.js --save <audio ref="audio"></audio> import H ...

  7. Netty 面试题 (史上最全、持续更新)

    文章很长,建议收藏起来,慢慢读! 疯狂创客圈为小伙伴奉上以下珍贵的学习资源: 疯狂创客圈 经典图书 : <Netty Zookeeper Redis 高并发实战> 面试必备 + 大厂必备 ...

  8. 屌炸天,像写代码一样写PPT,一个小工具解决

    此文已经废,请移步升级版博文: markdown写ppt (史上最全)

  9. 【题解】codeforces 467C George and Job dp

    题目描述 新款手机 iTone6 近期上市,George 很想买一只.不幸地,George 没有足够的钱,所以 George 打算当一名程序猿去打工.现在George遇到了一个问题. 给出一组有 n ...

  10. SCP,SSH应用