Java 读写锁实现原理

2018 年 8 月 16 日 开源中国

#扫描上方二维码报名成都源创会#


作者:牛李

链接:

https://my.oschina.net/editorial-story/blog/1928306

本文为作者投稿文章,转载请注明上述信息


最近做的一个小项目中有这样的需求:整个项目有一份config.json保存着项目的一些配置,是存储在本地文件的一个资源,并且应用中存在读写(读>>写)更新问题。既然读写并发操作,那么就涉及到操作互斥,这里自然想到了读写锁,本文对读写锁方面的知识做个梳理。


为什么需要读写锁?


与传统锁不同的是读写锁的规则是可以共享读,但只能一个写,总结起来为:读读不互斥,读写互斥,写写互斥,而一般的独占锁是:读读互斥,读写互斥,写写互斥,而场景中往往读远远大于写,读写锁就是为了这种优化而创建出来的一种机制。


注意是读远远大于写,一般情况下独占锁的效率低来源于高并发下对临界区的激烈竞争导致线程上下文切换。因此当并发不是很高的情况下,读写锁由于需要额外维护读锁的状态,可能还不如独占锁的效率高。因此需要根据实际情况选择使用。


一个简单的读写锁实现


根据上面理论可以利用两个int变量来简单实现一个读写锁,实现虽然烂,但是原理都是差不多的,值得阅读下。



ReadWriteLock的实现原理


在Java中ReadWriteLock的主要实现为ReentrantReadWriteLock,其提供了以下特性:


  1. 公平性选择:支持公平与非公平(默认)的锁获取方式,吞吐量非公平优先于公平。

  2. 可重入:读线程获取读锁之后可以再次获取读锁,写线程获取写锁之后可以再次获取写锁

  3. 可降级:写线程获取写锁之后,其还可以再次获取读锁,然后释放掉写锁,那么此时该线程是读锁状态,也就是降级操作。


ReentrantReadWriteLock的结构


ReentrantReadWriteLock的核心是由一个基于AQS的同步器Sync构成,然后由其扩展出ReadLock(共享锁),WriteLock(排它锁)所组成。



并且从ReentrantReadWriteLock的构造函数中可以发现ReadLock与WriteLock使用的是同一个Sync,具体怎么实现同一个队列既可以为共享锁,又可以表示排他锁下文会具体分析。


清单一:ReentrantReadWriteLock构造函数



Sync的实现


sync是读写锁实现的核心,sync是基于AQS实现的,在AQS中核心是state字段和双端队列,那么一个一个问题来分析。


Sync如何同时表示读锁与写锁?


清单2:读写锁状态获取



从代码中获取读写状态可以看出其是把state(int32位)字段分成高16位与低16位,其中高16位表示读锁个数,低16位表示写锁个数,如下图所示(图来自Java并发编程艺术)。


 



该图表示当前一个线程获取到了写锁,并且重入了两次,因此低16位是3,并且该线程又获取了读锁,并且重入了一次,所以高16位是2,当写锁被获取时如果读锁不为0那么读锁一定是获取写锁的这个线程。


读锁的获取


读锁的获取主要实现是AQS中的acquireShared方法,其调用过程如下代码。


清单3:读锁获取入口



其中doAcquireShared(arg)方法是获取失败之后AQS中入队操作,等待被唤醒后重新获取,那么关键点就是tryAcquireShared(arg)方法,方法有点长,因此先总结出获取读锁所经历的步骤,获取的第一部分步骤如下:


  • 操作1:读写需要互斥,因此当存在写锁并且持有写锁的线程不是该线程时获取失败。

  • 操作2:是否存在等待写锁的线程,存在的话则获取读锁需要等待,避免写锁饥饿。(写锁优先级是比较高的)

  • 操作3:CAS获取读锁,实际上是state字段的高16位自增。

  • 操作4:获取成功后再ThreadLocal中记录当前线程获取读锁的次数。


清单4:读锁获取的第一部分



当操作2,操作3失败时会执行fullTryAcquireShared(current),为什么会这样写呢?个人认为是一种补偿操作,操作2与操作3失败并不代表当前线程没有读锁的资格,并且这里的读锁是共享锁,有资格就应该被获取成功,因此给予补偿获取读锁的操作。在fullTryAcquireShared(current)中是一个循环获取读锁的过程,大致步骤如下:


  • 操作5:等同于操作2,存在写锁,且写锁线程并非当前线程则直接返回失败

  • 操作6:当前线程是重入读锁,这里只会偏向第一个获取读锁的线程以及最后一个获取读锁的线程,其他都需要去AQS中排队。

  • 操作7:CAS改变读锁状态

  • 操作8:同操作4,获取成功后再ThreadLocal中记录当前线程获取读锁的次数。


清单5:读锁获取的第二部分



读锁的释放


清单6:读锁释放入口



读锁的释放主要是tryReleaseShared(arg)函数,因此拆解其步骤如下:


  • 操作1:清理ThreadLocal中保存的获取锁数量信息

  • 操作2:CAS修改读锁个数,实际上是自减一


清单7:读锁的释放流程



写锁的获取


清单8:写锁的获取入口



写锁的获取也主要是tryAcquire(arg)方法,这里也拆解步骤:


  • 操作1:如果读锁数量不为0或者写锁数量不为0,并且不是重入操作,则获取失败。

  • 操作2:如果当前锁的数量为0,也就是不存在操作1的情况,那么该线程是有资格获取到写锁,因此修改状态,设置独占线程为当前线程


清单9:写锁的获取



写锁的释放


清单10:写锁的释放入口



写锁的释放主要是tryRelease(arg)方法,其逻辑就比较简单了,注释很详细。


清单11:写锁的释放



一些其他问题


锁降级操作哪里体现?


锁降级操作指的是一个线程获取写锁之后再获取读锁,然后读锁释放掉写锁的过程。在tryAcquireShared(arg)获取读锁的代码中有如下代码。


清单12:写锁降级策略



那么锁降级有什么用?答案是为了可见性的保证。在ReentrantReadWriteLock的javadoc中有如下代码,其是锁降级的一个应用示例。



公平与非公平的区别


清单13:公平下的Sync



公平下的Sync实现策略是所有获取的读锁或者写锁的线程都需要入队排队,按照顺序依次去尝试获取锁。


清单14:非公平下的Sync



非公平下由于抢占式获取锁,写锁是可能产生饥饿,因此解决办法就是提高写锁的优先级,换句话说获取写锁之前先占坑。

 

作者:牛李,一个正在努力学习的码农,主要关注后端领域、代码设计,以及一些有趣的技术。


GitHub: https://github.com/mrdear


本文系作者投稿文章。欢迎投稿。


投稿方式


  • 以 Word 或者 Markdown 文档的形式将稿件投递到 oscbianji@oschina.cn 邮箱


好书大放送!


最新 Spring Cloud+ Spring Boot 2.0 技术书免费送!


参与留言:

实力送书!你真的懂 Spring Cloud 微服务吗?

获赞最多的前6名就能把书抱回家!

活动时间:8月11日-18日





推荐阅读

你期待的 Win 10+Chrome OS 双系统正在研发中

VS Code 1.26 发布,有你想要的新特性?

集 Python、C、R 与 Ruby 之长,Julia 1.0 发布

Linux 命令行厉害,其实 Windows 的也很强

微软按月收费桌面计划,Win 10 将变成 Win 365?

点击“阅读原文”查看更多精彩内容

登录查看更多
0

相关内容

读远是一个公益的电子书分享与阅读社区,主要面向Kindle、iPad等移动阅读设备用户,致力于通过社区的力量为读远用户提供优质的阅读内容与服务。
【2020新书】使用高级C# 提升你的编程技能,412页pdf
专知会员服务
58+阅读 · 2020年6月26日
【IJCAI2020-华为诺亚】面向深度强化学习的策略迁移框架
专知会员服务
28+阅读 · 2020年5月25日
【圣经书】《强化学习导论(2nd)》电子书与代码,548页pdf
专知会员服务
203+阅读 · 2020年5月22日
【经典书】数据结构与算法C++,第二版,738页pdf
专知会员服务
168+阅读 · 2020年3月27日
《深度学习》圣经花书的数学推导、原理与Python代码实现
【书籍推荐】简洁的Python编程(Clean Python),附274页pdf
专知会员服务
181+阅读 · 2020年1月1日
【电子书】Flutter实战305页PDF免费下载
专知会员服务
23+阅读 · 2019年11月7日
告别 PS !3 行代码 5 秒搞定抠图的 AI 神器!
程序人生
6+阅读 · 2019年7月11日
使用 Canal 实现数据异构
性能与架构
20+阅读 · 2019年3月4日
Flink 靠什么征服饿了么工程师?
阿里技术
6+阅读 · 2018年8月13日
Python 杠上 Java、C/C++,赢面有几成?
CSDN
6+阅读 · 2018年4月12日
YOLO算法的原理与实现
机器学习研究会
43+阅读 · 2018年1月19日
循环神经网络的介绍、代码及实现
AI研习社
3+阅读 · 2017年11月21日
相似图片搜索的原理
数据库开发
9+阅读 · 2017年8月11日
最大熵原理(一)
深度学习探索
12+阅读 · 2017年8月3日
A Sketch-Based System for Semantic Parsing
Arxiv
4+阅读 · 2019年9月12日
Knowledge Based Machine Reading Comprehension
Arxiv
4+阅读 · 2018年9月12日
Arxiv
6+阅读 · 2018年8月27日
Arxiv
3+阅读 · 2018年5月28日
Arxiv
3+阅读 · 2018年4月18日
Arxiv
5+阅读 · 2018年1月30日
VIP会员
相关VIP内容
【2020新书】使用高级C# 提升你的编程技能,412页pdf
专知会员服务
58+阅读 · 2020年6月26日
【IJCAI2020-华为诺亚】面向深度强化学习的策略迁移框架
专知会员服务
28+阅读 · 2020年5月25日
【圣经书】《强化学习导论(2nd)》电子书与代码,548页pdf
专知会员服务
203+阅读 · 2020年5月22日
【经典书】数据结构与算法C++,第二版,738页pdf
专知会员服务
168+阅读 · 2020年3月27日
《深度学习》圣经花书的数学推导、原理与Python代码实现
【书籍推荐】简洁的Python编程(Clean Python),附274页pdf
专知会员服务
181+阅读 · 2020年1月1日
【电子书】Flutter实战305页PDF免费下载
专知会员服务
23+阅读 · 2019年11月7日
相关资讯
告别 PS !3 行代码 5 秒搞定抠图的 AI 神器!
程序人生
6+阅读 · 2019年7月11日
使用 Canal 实现数据异构
性能与架构
20+阅读 · 2019年3月4日
Flink 靠什么征服饿了么工程师?
阿里技术
6+阅读 · 2018年8月13日
Python 杠上 Java、C/C++,赢面有几成?
CSDN
6+阅读 · 2018年4月12日
YOLO算法的原理与实现
机器学习研究会
43+阅读 · 2018年1月19日
循环神经网络的介绍、代码及实现
AI研习社
3+阅读 · 2017年11月21日
相似图片搜索的原理
数据库开发
9+阅读 · 2017年8月11日
最大熵原理(一)
深度学习探索
12+阅读 · 2017年8月3日
Top
微信扫码咨询专知VIP会员