RWLock——一种细粒度的Mutex互斥锁
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)
}
}
如果锁状态已经是解锁状态则抛异常。
如果是只剩下一个读等待,则释放写信号量通知其他正在尝试持有写锁的协程上锁。
关于信号量的细节
我们上面分析了读写锁的上锁与解锁的过程,其实有一个点不知道大家有没有注意。就是关于信号量的操作对象的细节。
- 调用 Lock 获取写锁,会持有 writerSem 信号
- 调用 Unlock 释放写锁时,会释放 readerSem 信号
- 调用 RLock 获取读锁时,会持有 readerSem 信号
- 调用 RUnlock 释放读锁时,会释放 writerSem 信号
大家有没有发现其中的规律,这么做的目的是什么呢?
也就是说:我们在获取写锁之前,会先等待读锁的释放操作。而在获取读锁时,会先等待写锁的释放操作。
我们用反证法来假设这个场景:我这里有一个连续的写操作;那么也就是说我要连续反复的调用 Lock + Unlock 操作。如果没有上面的信号量的互相牵制,那么就很容易出现读操作没法执行的问题,也就是说会”饿死“。
所以 RWMutex 加入读写信号量的机制是为了更好达到 RW 的目的,而不是一直 W。
总结
- 在调用 Lock 获取写锁时,会先等待 RUnlock 将其 readerCount 置为 0,然后成功获取写锁。
- 还有一个操作是将 readerCount - rwmutexMaxReaders,其目的是为了阻塞后续的 RLock 操作。即在读取写锁其他任何读写操作都不允许了。
- 在调用 Unlock 释放写锁时,会通知所有读操作,解锁那些阻塞的读锁,然后成功释放写锁。
RWLock——一种细粒度的Mutex互斥锁的更多相关文章
- golang mutex互斥锁分析
互斥锁:没有读锁写锁之分,同一时刻,只能有一个gorutine获取一把锁 数据结构设计: type Mutex struct { state int32 // 将一个32位整数拆分为 当前阻塞的gor ...
- Go 标准库 —— sync.Mutex 互斥锁
Mutex 是一个互斥锁,可以创建为其他结构体的字段:零值为解锁状态.Mutex 类型的锁和线程无关,可以由不同的线程加锁和解锁. 方法 func (*Mutex) Lock func (m *Mut ...
- C# Mutex互斥锁
Mutex 构造函数 (Boolean, String, Boolean) public Mutex ( bool initiallyOwned, string name, out bool crea ...
- C# mutex互斥锁构造
概念 Mutext 出现的比monitor更早,而且传承自COM,当然,waitHandle也是它的父类,它继承了其父类的功能,有趣的是Mutex的脾气非常的古怪,它 允许同一个线程多次重复访问共享区 ...
- 【转】【C#】【Thread】Mutex 互斥锁
Mutex:互斥(体) 又称同步基元. 当创建一个应用程序类时,将同时创建一个系统范围内的命名的Mutex对象.这个互斥元在整个操作系统中都是可见的.当已经存在一个同名的互斥元时,构造函数将会输出一个 ...
- linux c 线程间同步(通信)的几种方法--互斥锁,条件变量,信号量,读写锁
Linux下提供了多种方式来处理线程同步,最常用的是互斥锁.条件变量.信号量和读写锁. 下面是思维导图: 一.互斥锁(mutex) 锁机制是同一时刻只允许一个线程执行一个关键部分的代码. 1 . ...
- Golang 读写锁RWMutex 互斥锁Mutex 源码详解
前言 Golang中有两种类型的锁,Mutex (互斥锁)和RWMutex(读写锁)对于这两种锁的使用这里就不多说了,本文主要侧重于从源码的角度分析这两种锁的具体实现. 引子问题 我一般喜欢带着问题去 ...
- 探索互斥锁 Mutex 实现原理
Mutex 互斥锁 概要描述 mutex 是 go 提供的同步原语.用于多个协程之间的同步协作.在大多数底层框架代码中都会用到这个锁. mutex 总过有三个状态 mutexLocked: 表示占有锁 ...
- 互斥锁Mutex与信号量Semaphore的区别
转自互斥锁Mutex与信号量Semaphore的区别 多线程编程中,常常会遇到这两个概念:Mutex和Semaphore,两者之间区别如下: 有人做过如下类比: Mutex是一把钥匙,一个人拿了就可进 ...
随机推荐
- typora的一些使用
1.介绍typora 支持markdown语法的一款写作app 真的足够简洁高效 2. typora和其他工具配合实现功能 如插入图片 截图 gif等等图库 smms图库的使用 需要使用PicGo和s ...
- LATEX如何写多个条件推导式推出一个结论
用markdown写推导式的时候,不太好写,故做个笔记记录一下,插入 公式块 : \left. \begin{aligned} \left. \begin{aligned} \text{这里可以写文字 ...
- Go string 详解
前言 字符串(string) 作为 go 语言的基本数据类型,在开发中必不可少,我们务必深入学习一下,做到一清二楚. 本文假设读者已经知道切片(slice)的使用,如不了解,可阅读 Go 切片 基本知 ...
- MegEngine基本概念
MegEngine基本概念 基本概念 MegEngine 是基于计算图的深度神经网络学习框架. 本文内容会简要介绍计算图及其相关基本概念,以及在 MegEngine 中的实现. 计算图 结合一个简单的 ...
- 【Java面试真题】剑指Offer53.2——0~n-1中缺失的数字(异或、二分两种解法)
[Java实现]剑指Offer53.2--0~n-1中缺失的数字:面试真题,两种思路分享 前面有另一道面试题[Java实现]剑指offer53.1--在排序数组中查找数字(LeetCode34:在排序 ...
- 基本FTP服务: 文件传输协议
1.服务端虚拟机Server,安装可以提供FTP服务软件 [root@server0 ~]# yum -y install vsftpd 2.虚拟机Server操作,启动 vsftpd服务,设置为开机 ...
- CSS基础知识及其基本语法
一.什么是CSS CSS 是层叠样式表( Cascading Style Sheets ) 的简称. 有时我们也会称之为CSS 样式表或级联样式表. CSS 也是一种标记语言 CSS 主要用于设置HT ...
- 基于Docker安装常用软件
基于Docker安装常用软件 本实验介绍如何基于Docker安装常用的软件,具体包括: Ubuntu Cetnos Nginx Node.js PHP MySQL Tomcat Redis Mongo ...
- [Linux]经典面试题 - 网络基础 - TCP三次握手
[Linux]经典面试题 - 网络基础 - TCP三次握手 目录 [Linux]经典面试题 - 网络基础 - TCP三次握手 一.TCP报文格式 1.1 TCP报头 1.2 报文图例 二.TCP三次握 ...
- Python-对比两个目录中Excel文件
背景:我在5月20日收到了一批Excel文件数据,由于文件很多大约有将近5000个,已经通过编写python脚本处理完成.但是6月9日的时候,又收到了一批新的Excel数据.但是在处理过程中发现,本次 ...