Java多线程(二)——ReentrantLock源码解析(补充1——从AQS中唤醒的线程)

news2024/11/15 12:42:35

ReentrantLock源码解析(补充1)

上一章仅介绍了 ReentrantLock 的常用方法以及公平锁、非公平锁的实现。这里对上一章做一些补充。主要是:

  • AQS 中阻塞的线程被唤醒后的执行流程 (本篇讲述)

  • 可打断的锁 lock.lockInterruptibly()

  • 锁超时 lock.tryLock(long,TimeUnit)

  • 条件变量 Condition

1. AQS 中阻塞的线程被唤醒后的执行流程

线程尝试获取锁,不论公平锁还是非公平锁,如果获取不到,最后都会进入到 AQS 的队列中进行阻塞等待。直到持有锁的线程将锁释放,通过从后往前遍历查找 AQS 队列中距离 head 最近的可用节点,将其线程唤醒:LockSupport.unpark( s.thread)。 唤醒后的线程将会继续尝试竞争锁,我们分析一下它是如何竞争的:

1.1 线程被唤醒后,会继续执行 LockSupport.park() 之后的代码

LockSupport.park(thread) 方法会让Java线程进入 等待状态(WAITING),Java线程状态详情参见:

Java-线程基础

LockSupport.unpark(thread)调用之后,线程被唤醒,并获取到 CPU 时间片后,将继续运行 LockSupport.park() 位置后续的代码。(thread.interrupt()也可以将在 等待状态 线程唤醒)

在 ReentrantLock 中,AQS 中阻塞等待的线程被唤醒后,将继续下述代码的内容:

//AbstractQueuedSynchronized.java
final boolean acquireQueued(final Node node, int arg) {
    boolean failed = true;
    try {
        boolean interrupted = false;
        for (;;) {
            final Node p = node.predecessor();
            //如果当前节点的前驱为head(这是一个空节点,标志着 AQS 双向链表中的头),那么可以尝试获取锁
            if (p == head && tryAcquire(arg)) {
                setHead(node);
                p.next = null; // help GC
                failed = false;
                return interrupted;
            }

            //之前被阻塞在 parkAndCheckInterrupt() 方法中
            //shouldParkAfterFailedAcquire()方法将node的前驱节点中废弃节点(waitStatus = CANCELLED = 1)全都清理出队列。
            //1. 如果有废弃节点要清理,那么该方法return false,视图再次进入循环判断当前线程所在节点是否处在队列的队头位置。
            //2. 如果没有废弃节点要清理,那么说明当前线程不处在队头,那么就要进入 等待状态(WAITING) 阻塞。
            if (shouldParkAfterFailedAcquire(p, node) &&
                parkAndCheckInterrupt())
                interrupted = true;
        }
    } finally {
        if (failed)
            cancelAcquire(node);
    }
}

private final boolean parkAndCheckInterrupt() {
    //之前通过LockSupport.park()方法进入阻塞状态
    //LockSupport.unpark()或者interrupt()之后,会继续代码执行
    LockSupport.park(this);
    //Thread.interrupted()将会返回线程的打断标记,并且清空打断标记。
    return Thread.interrupted();
}

由于 LockSupport.park() 而处在 等待状态(WAITING) 的线程不仅可以通过 LockSupport.unpark() 方法唤醒,也可以通过 thread.interrupt() 方法唤醒,区别是后者将会让 thread 的打断标记置为 true。

线程唤醒后,继续执行 parkAndCheckInterrupt() 的 return 部分代码。而后进入到acquireQueued() 中继续死循环,尝试获取锁,或者重新回到 AQS 的等待队列中阻塞。

1.2 线程唤醒后的竞争分析

1. 解锁前:(node为双向指针,我画漏了)

假设thread1所在node之前所有的废弃节点(waitStatus = CANCELLED = 1)以及全部清空,当前 node(thread1) 已经处在队列头。

请添加图片描述

被唤醒时,进入 acquireQueued() 的 for(;😉 循环,由于 node(thread1) 为队头,由 acquireQueued() 代码可知,该线程将进行 tryAcquire() 尝试获取锁资源。

final boolean acquireQueued(final Node node, int arg) {
	...
    final Node p = node.predecessor();
    if (p == head && tryAcquire(arg)) {
        ...
    }
	...
}

如果没有出现竞争,如上一章讨论的上锁流程, node(thread1) 将会成为新的空head,thread1 从 node 中脱离出来,继续执行后续临界区代码。

本篇中,我们补充讨论出现竞争的情况:

2. 解锁后,出现竞争:(node为双向指针,我画漏了)

请添加图片描述

node(thread1) 被唤醒后,由于处在空head之后,为首个可用队头节点,将进入到 tryAcquire() 尝试获取所资源,同时 thread3 也在尝试获取锁资源。

3. 竞争失败 (node为双向指针,我画漏了)

请添加图片描述

如果 node(thread1) 竞争失败,将会进入 acquireQueue() 中下列部分,再次通过 parkAndCheckInterrupt() 的 LockSupport.lock() 方法阻塞起来。需要注意的是,这里没有额外动作,node(thread1) 在队列中的位置不变:

final boolean acquireQueued(final Node node, int arg) {
    try {
        for (;;) {
            final Node p = node.predecessor();
            if (p == head && tryAcquire(arg)) {
               ...
            }
            // tryAquire()竞争失败 return false后,进入下面部分
            if (shouldParkAfterFailedAcquire(p, node) &&
                //再次阻塞起来
                parkAndCheckInterrupt())
                interrupted = true;
        }
    } finally {
        if (failed)
            cancelAcquire(node);
    }
}

4.竞争成功

请添加图片描述

竞争成功,thread1将会从node中解放出来,进入临界区运行后续代码:

Thread thread1 = new Thread(()->{
   lock.lock();
    //获取到锁之后,进入下面的临界区代码
   try{
       //临界区代码
   }finally{
       lock.unlock();
   }
});

而 node 将会被置空,并成为新的 空head 节点,原先的 空head 节点被抛弃:

final boolean acquireQueued(final Node node, int arg) {
    ...
    if (p == head && tryAcquire(arg)) {
        //争锁成功,成为新的 空head
        setHead(node);
        //将原先的空head抛弃
        p.next = null; // help GC
        failed = false;
        return interrupted;
    }
    ...     
}

private void setHead(Node node) {
    //成为新的head
    head = node;
    //将封装在 node 中的 thread 解放出去
    node.thread = null;
    //清空prev指向
    node.prev = null;
}

而竞争失败的 thread3 将会如上一篇所言,进入 AQS 等待队列,并通过 LockSupport.park() 进入 等待状态(WAITING)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/176847.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【QT5.9】与MFC对比学习笔记-感悟篇2【2023.01.23】

是对QT的分析,不仅局限于QT。 二者区别 天下文章一大抄,技术也一样。MFC是对Windows系统API进行的封装,是以视类与文档类为核心的框架设计。微软20年前就已经把MVC玩的很6了,还有控件、动态库等等技术都是微软爸爸先搞出来的。若…

Kubernetes:认识 K8s开源 Web/桌面 客户端工具 Headlamp

写在前面 分享一个 k8s 客户端开源项目 Headlamp 给小伙伴博文内容涉及: Headlamp 桌面/集群 Web 端安装启动导入集群简单查看集群信息 理解不足小伙伴帮忙指正 我所渴求的,無非是將心中脫穎語出的本性付諸生活,為何竟如此艱難呢 ------赫尔曼…

第八层:模板

文章目录前情回顾模板模板的概念模板的特点模板分类函数模板作用语法函数模板的使用注意事项普通函数和函数模板的区别普通函数和函数模板的调用规则优先调用普通函数空模板强调函数模板函数模板可以发生重载函数模板产生更好的匹配时模板的局限性类模板作用语法类模板实例化对…

Redis在秒杀场景的作用

秒杀业务特点:限时限量,业务系统要处理瞬时高并发请求,Redis是必需品。 秒杀可分成秒杀前、秒杀中和秒杀后三阶段,每个阶段的请求处理需求不同,Redis具体在秒杀场景的哪个环节起到作用呢? 1 秒杀负载特征…

Java-数据结构-二叉树<三>

承接上文: Java-数据结构-二叉树<一> Java-数据结构-二叉树<二> 一. 二叉树的简单介绍 见Java-数据结构-二叉树<一> 二. 二叉树的典型代码实现 见Java-数据结构-二叉树<一&#x…

4. RNN网络架构解读|词向量模型|模型整体框架|训练数据构建|CBOW和Skip-gram模型|负采样方案

文章目录RNN网络架构解读词向量模型模型整体框架训练数据构建CBOW和Skip-gram模型负采样方案RNN网络架构解读 递归神经网络实际上就是普通的神经网络的部分进行修改更新:实际上常用于时间序列的更新。或者就是自然处理中 X序列代表着时间序列,x0是一个时…

linux入门---云服务器购买和登陆

目录标题云服务器选择云服务器购买xshell下载如何登陆云服务器Linux的新建与删除新建删除云服务器选择 学习linux的时候云服务器是一个非常重要的工具,那么我们在购买云服务器的时候有很多选择比如说:华为云,腾讯云,阿里云等等&a…

【实操案例十二】类和对象 实例代码及运行效果图!

任务一:定义一个圆的类,计算面积和周长 # 任务一:定义一个圆的类,计算面积和周长 import math class Circle():def __init__(self,r):self.rrdef get_area(self):return math.pi*r*rdef get_perimeter(self):return 2*math.pi*r …

初识 ThreeJS (ThreeJS 相关环境搭建)

初识 ThreeJS (初识 ThreeJS (ThreeJS 相关环境搭建)参考描述ThreeJS在本地搭建 NodeJS 的官方网站获取使用安装依赖项运行官方文档案例场景编辑器搭建 ThreeJS 运行环境webpack项目结构package.jsonwebpack.config.js深入获取检测参考 项目…

袋式除尘器—分类和命名

按除尘器的结构形式分类(1)按滤袋开头分类按滤袋形状分类,可分为圆袋式除尘器和扁袋式除尘器两类。①圆袋式除尘器。滤袋形状为圆筒形,直径一般为120~300mm,最大不超过600mm;高度为2~3m,也有10m…

redis 数据库简介

一 概述 redis是一种nosql数据库,他的数据是保存在内存中,同时redis可以定时把内存数据同步到磁盘,即可以将数据持久化,并且他比memcached支持更多的数据结构(string,list列表[队列和栈],set[集合],sorted set[有序集合],hash(hash表))。相关…

2023年哪款手机浏览器比较好用,最后一个吹爆它

很多人不满足于手机自带的浏览器,为了更好地满足看视频、浏览网页、看小说等需求,不少人下载第三方手机浏览器来使用。我们都知道,手机浏览器是手机不可缺少的APP之一。那么,2023年哪款手机浏览器比较好用?下面分享今年…

Java File类及案例

File概述和构造方法 File对象就表示一个路径,可以是文件路径、也可以是文件夹的路径这个路径可以是存在的,也允许是不存在的 方法名称说明public File (String pathname)把字符串表示的路径变成File对象public File (Srting parent, String child)把父…

【My Electronic Notes系列——三极管】

目录 序言: 🏮🏮新年的钟声响,新年的脚步迈,祝新年的钟声,敲响你心中快乐的音符,幸运与平安,如春天的脚步紧紧相随,春节快乐!春华秋实,我永远与你…

C语言入门(八)——数组

数组的基本概念 数组应用实例:统计随机数 数组应用实例:直方图 字符串 多维数组 数组的基本概念 数组(Array)也是一种复合数据类型,它由一系列相同类型的元素(Element)组成。例如定义一个由4个int型元素组成的数组count: int count[4]; 和结构体成员类似&…

安卓S开机动画流程

安卓S开机动画流程 开机动画是在SurfaceFlinger实例通过调用startBootAnim()启动的,BootAnim是如何启动和结束的,总体框架图如下: 1.SurfaceFlinger进程启动 # /frameworks/native/services/surfaceflinger/surfaceflinger.rc service surf…

linux inode详解

1.inode 和 block 概述. 操作系统的文件数据除了实际内容之外,通常含有非常多的属性,例如Linux操作系统的文件权限与文件属性。文件系统通常会将这两部分内容分别存放在inode和block中。 文件是存储在硬盘上的,硬盘的最小存储单位叫做扇区sec…

行为型模式

1.模版方法 超类中定义了一个算法的框架, 允许子类在不修改结构的情况下重写算法的特定步骤 结构 抽象类:负责给出一个轮廓与骨架,由一个模版方法和若干个基本方法构成 模版方法:按某种顺序调用其包含的基本方法基本方法&#xf…

计算机视觉OpenCv学习系列:第八部分、图像操作-4

第八部分、图像操作-4第一节、图像卷积操作1.图像卷积定义2.卷积函数3.代码练习与测试第二节、高斯模糊1.高斯模糊2.函数解释3.代码练习与测试第三节、像素重映射1.像素重映射定义2.重映射函数3.代码练习与测试学习参考第一节、图像卷积操作 1.图像卷积定义 卷积的基本原理&am…

java spring IOC xml 方式 内部Bean注入

上次说了外部 Bean注入 这次来演示一个内部的 Bean注入 我们先创建一个spring 项目 导入最基本的 spring 包 在项目src目录下创建一个包 cascade cascade包下创建两个类 Dept 部门类 参考代码如下 package cascade;//部门类 public class Dept {private String dname;publi…