1. 问题

最近有同事问了我一个问题,在Java编程中,当有一条线程要获取ReentrantReadWriteLock的读锁,此时已经有其他线程获得了读锁,AQS队列里也有线程在等待写锁。由于读锁是共享锁,当前线程是马上获得读锁,还是排队?如果是马上获得读锁,那岂不是阻塞的等待写锁的线程有可能一直(或长时间)拿不到写锁(写锁饥饿)?

带着这个问题,我打开读写锁的源码,来看一下JDK是怎么实现的。(注:读写锁指ReentrantReadWriteLock, 以下说到的读锁和写锁,都是指属于同一个读写锁的情况。读锁和共享锁,写锁和独占锁,在这里是同样的意思。如无特殊说明,提到的模式都是默认的非公平模式)

2. JUC万物皆有AQS

2.1 读锁的实现。

先来看看读锁的实现。持有一个AQS,所以说,JUC万物皆有AQS(大雾)。

顺便提一下写锁,写锁也是类似的实现,而且传入的是同一个读写锁,那么读锁和写锁,都拥有同一个AQS,这样才能实现互相阻塞。

读锁是共享模式。

2.2 tryAcquireShared(int arg)的实现。

熟悉AQS的同学就知道,共享锁的实现,AQS已经写好了流程。但留下了一个钩子,tryAcquireShared(int arg) 供各种场景实现。

那么我们就来看看,读写锁里面,共享锁(读锁)是怎么实现的。

step1. 红框一,如果当前已经有线程持有了独占锁(即写锁),且不是当前线程持有,那么无法重入,直接返回-1,获取共享锁失败。

step2. 如果step1的情况被排除,那么进行readerShouldBlock()的判断。在读写锁中,AQS有两种实现,公平和非公平模式,默认是非公平模式。

也就是说,上面所说的sync变量的实际类型,可以是公平模式,也可以是非公平模式。

因此,readerShouldBlock()也有公平和非公平两种不同的实现。

公平模式下,只要前面有阻塞排队的节点,就返回true,表示不能抢占。

非公平模式下,看看第一个等待的阻塞节点是不是独占式的,如果是,返回true,有可能不可以抢在人家前面(为什么是有可能?要考虑可重入的场景,下面分析)。这是为了避免写锁饥饿。

所以,如果readerShouldBlock()返回false,并且读锁获取的总次数不溢出,且CAS成功,说明获取共享锁成功,下面进入if块,设置一些变量,并将当前线程持有的该读锁的次数递增加1,返回成功标志。

看到这里,也许你会有疑惑,仅仅是因为CAS失败,就获取共享锁失败了吗?而且,ReentrantReadWriteLock是一个可重入锁,这里也没看到有重入的地方啊。

别急,如果step2失败,会进入step3,到第三个红框,进入fullTryAcquireShared(Thread current)方法。

2.3  final int fullTryAcquireShared(Thread current)

这个方法比较长,里面用了for(;;) 自旋CAS,为什么呢?因为CAS还是可能会失败啊……失败就得继续再尝试一把。

我就贴出for(;;) 里的代码,分为两段,第一段判断是否可以尝试获取锁(与上面类似,加了重入的判断),第二段CAS和成功后的一些操作。

先看第一段,判断是否可以尝试获取锁。

step1. 如果有线程持有独占锁,并且不是当前线程,返回失败标志-1。如果是当前线程,由于可重入的语义,通过了判断,直接跑到第二段代码了。说明在持有独占锁的情况下可以获取共享锁(锁降级)。

step2. 如果当前没有线程持有独占锁,那么再来看看熟悉的readerShouldBlock()。通过上面的分析我们知道,在公平模式下有节点在阻塞就得排队,在非公平模式下有可能不可以抢在人家前面。为什么是有可能?因为要考虑可重入的场景。

如果firstReader是当前线程,或者当前线程的cachedHoldCounter变量的count不为0(表示当前线程已经持有了该共享锁),均说明当前线程已经持有共享锁,此次获取共享锁是重入,这也是允许的,可以通过判断。

如果可以顺利通过上面两步判断,说明获取共享锁成功,下面开始熟悉的CAS。

失败了咋办?别忘记是自旋啊,外层是for(;;),那就再来一发~~。当然还得再来一遍第一段的判断。

3. 结论

经过上面的分析,可以来回答我的同事的问题了。

在Java编程中,当有一条线程要获取ReentrantReadWriteLock的读锁,此时已经有其他线程获得了读锁,AQS队列里也有线程在等待写锁。由于读锁是共享锁,当前线程是马上获得读锁,还是排队?如果是马上获得读锁,那岂不是阻塞的等待写锁的线程有可能一直(或长时间)拿不到写锁(写锁饥饿)?

1.如果已经有线程持有独占锁

1.1 该线程不是当前线程,不用想了,乖乖排队;

1.2 该线程就是当前线程,重入,CAS获取共享锁;

2.如果没有线程持有独占锁,检查当前线程是否需要block(readerShouldBlock方法)。

block的判断,有两种模式,公平和非公平(默认模式)。如果不需要block, 必须满足:公平模式下,没有节点在AQS等待;非公平模式下,AQS第一个等待的节点不是独占式的;

2.1 不需要block,可以CAS获取共享锁;

2.2 需要block;

2.2.1 当前线程已经持有了共享锁,重入,还是可以CAS获取共享锁;

2.2.2 当前线程前没有已经持有共享锁,则获取失败,只能排队。

上面是根据代码逻辑整理的,可以换为更简洁的语言。

如果当前线程已经持有独占锁或共享锁(重入)或不需要block,则CAS获取共享锁;否则,排队。

readerShouldBlock()判断第一个节点是获取共享锁或独占锁,在不考虑重入的情况下,是什么意思呢?

1. 第一个节点是等待独占锁的场景,说明下一个就是它了,不能抢它的,抢不到;

2. 第一个节点是等待共享锁的场景,说明第一个节点,

2.1 在等待持有独占锁的线程释放独占锁,这种必然是抢不到的。

2.2 持有共享锁的线程还在唤醒后续节点的过程中,允许你去抢一下。当然,不意味着一定可以抢成功。

如果是2.2持有共享锁的线程在唤醒后续节点过程中,理论上是可能获取得到的。这种情况概率较小,我没重现过。

回到这个问题。当前线程并没有获取到写锁或读锁,不能重入;AQS中,第一个等待的大概率是想要获取独占锁的节点,必须block,所以当前线程只能排队,并不会出现阻塞的想获取写锁的节点一直拿不到写锁的情况;如果刚好没有完全唤醒,那么可能是可以抢占的。但也不会一直阻塞,因为唤醒节点获取读锁的过程是很快的。

总之,获取读锁的机制,记住这个结论就行。

如果当前线程已经持有独占锁或共享锁(重入)或不需要block,则CAS获取共享锁;否则,排队。

4. 举个栗子

第一个节点是独占锁的场景,不能抢占

 package com.khlin.my.test;

 import java.util.concurrent.locks.ReentrantReadWriteLock;

 public class RRWLockTest {

     public static void main(String[] args) throws InterruptedException {
final ReentrantReadWriteLock LOCK = new ReentrantReadWriteLock(); Thread reader1 = new Thread(new Runnable() {
public void run() {
try {
LOCK.readLock().lock();
System.out.println("reader1 locked.");
Thread.sleep(3000L);
System.out.println("reader1 finished.");
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
LOCK.readLock().unlock();
}
}
}); Thread reader2 = new Thread(new Runnable() {
public void run() {
try {
LOCK.readLock().lock();
System.out.println("reader2 locked.");
System.out.println("reader2 finished.");
} finally {
LOCK.readLock().unlock();
}
}
}); Thread writer = new Thread(new Runnable() {
public void run() {
try{
LOCK.writeLock().lock();
System.out.println("writer locked.");
System.out.println("writer finished.");
}finally {
LOCK.writeLock().unlock();
}
}
});
reader1.start();
Thread.sleep(1000L);
writer.start();
Thread.sleep(1000L);
reader2.start();
}
}

reader1获取了读锁,正在执行,随后writer来获取写锁,失败,入队等待。reader2由于writer正在等待(通过readerShouldBlock判断),无法获取读锁,入队,等待。输出如下:

简单分析线程获取ReentrantReadWriteLock 读锁的规则的更多相关文章

  1. [源码分析]读写锁ReentrantReadWriteLock

    一.简介 读写锁. 读锁之间是共享的. 写锁是独占的. 首先声明一点: 我在分析源码的时候, 把jdk源码复制出来进行中文的注释, 有时还进行编译调试什么的, 为了避免和jdk原生的类混淆, 我在类前 ...

  2. 简单分析ThreadPoolExecutor回收工作线程的原理

    最近阅读了JDK线程池ThreadPoolExecutor的源码,对线程池执行任务的流程有了大体了解,实际上这个流程也十分通俗易懂,就不再赘述了,别人写的比我好多了. 不过,我倒是对线程池是如何回收工 ...

  3. ffplay.c函数结构简单分析(画图)

    最近重温了一下FFplay的源代码.FFplay是FFmpeg项目提供的播放器示例.尽管FFplay只是一个简单的播放器示例,它的源代码的量也是不少的.之前看代码,主要是集中于某一个"点&q ...

  4. ffplay.c函数结构简单分析(绘图)

    近期重温了一下FFplay的源码. FFplay是FFmpeg项目提供的播放器演示样例.虽然FFplay不过一个简单的播放器演示样例,它的源码的量也是不少的. 之前看代码,主要是集中于某一个" ...

  5. SpringCloud配置刷新机制的简单分析[nacos为例子]

    SpringCloud Nacos 本文主要分为SpringCloud Nacos的设计思路 简单分析一下触发刷新事件后发生的过程以及一些踩坑经验 org.springframework.cloud. ...

  6. AbstractQueuedSynchronizer的简单分析

    说明:本作者是文章的原创作者,转载请注明出处:本文地址:http://www.cnblogs.com/qm-article/p/7955781.html 一.AbstractQueuedSynchro ...

  7. x264源代码简单分析:宏块分析(Analysis)部分-帧间宏块(Inter)

    ===================================================== H.264源代码分析文章列表: [编码 - x264] x264源代码简单分析:概述 x26 ...

  8. x264源代码简单分析:滤波(Filter)部分

    ===================================================== H.264源代码分析文章列表: [编码 - x264] x264源代码简单分析:概述 x26 ...

  9. x264源代码简单分析:编码器主干部分-1

    ===================================================== H.264源代码分析文章列表: [编码 - x264] x264源代码简单分析:概述 x26 ...

随机推荐

  1. 使用 acl_cpp 的 HttpServlet 类及服务器框架编写WEB服务器程序(系列文章)

    在 <用C++实现类似于JAVA HttpServlet 的编程接口 > 文章中讲了如何用 HttpServlet 等相关类编写 CGI 程序,于是有网友提出了 CGI 程序低效性,不错, ...

  2. 文件文件夹混合多选对话框(修改GWL_WNDPROC)

    /******************************************************************** created: 2008/07/22 created: 2 ...

  3. 基于ASP.NET的新闻管理系统(一)

    1. 项目简介 1.1设计内容 (1)可以在首页查看各类新闻,可以点击新闻查看具体内容:可以查看不同类型的新闻,并了解热点新闻,可以在搜索框里输入要查找的内容. (2)在后台界面中,管理员可以修改密码 ...

  4. WebApp 安全风险与防护课堂(第二讲)开课了!

    本文由葡萄城技术团队于原创并首发 转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者. 在昨天的公开课中,由于参与的小伙伴们积极性和热情非常高,我们的讲师Carl ...

  5. hgoi#20190514

    T1-Curriculum Vitae 给你一个长度为n的01序列a,删去其中的几个数,使得序列中左边是连续的0,右边是连续的1,可以没有0或1,求最多剩下几个数 解法 对于每个点看它左边几个0,右边 ...

  6. 快速理解类的访问控制(public,protected,private)

    接触过面向对象编程的朋友们都知道类中的访问控制符无非有三个,public, protected, private. 理解他就先从期望他达到的效果先说吧 Public: 使成员对于整个程序内(类内类外) ...

  7. 上不了名校?可以在 GitHub 上读他们的课程

    今天开始,全国各大区域的高考成绩陆续公布,又到了几家欢喜几家愁的时刻,如果你准备报考计算机相关专业,但是又由于分数不高而苦恼.别担心,在 GitHub 上有着大量的名校教学资源,即使上不了名校,也可以 ...

  8. Spark学习之路(七)—— 基于ZooKeeper搭建Spark高可用集群

    一.集群规划 这里搭建一个3节点的Spark集群,其中三台主机上均部署Worker服务.同时为了保证高可用,除了在hadoop001上部署主Master服务外,还在hadoop002和hadoop00 ...

  9. J2SE的基本简介与J2EE/J2ME的差异

    J2SE简介与J2EE.J2ME的比较 Java2平台包括:标准版(J2SE).企业版(J2EE)和微缩版(J2ME)三个版本. J2SE,J2ME和J2EE,这也就是SunONE(Open NetE ...

  10. RabbitMq-安装篇

    嘿,大家好,今天更新的内容是rabbitMq的安装篇~~ windows下安装rabbitMq rabbitMq下载地址:点我下载 1.由于rabbitMq用erlang语言开发,所以安装rabbit ...