AQS独占锁之ReentrantLock源码调试(JDK8)

news2024/12/27 14:18:58

前言:
为什么需要学习ReentrantLock?
目前项目开发中使用到的几乎都是分布式锁,平时可能很少用到java自带的锁; 但实际在我们java的源码中,随处可见需要使用锁来保证线程安全,所以还是有必要学习下ReentrantLock。

1.管程模型

可以说AQS就是基于管程模型来实现的,所以我们还需要了解;管程是管理共享变量以及对共享变量的操作过程,让他们支持并发。
在这里插入图片描述

在管程的发展史上,先后出现过三种不同的管程模型,分别是Hasen模型、Hoare模型和MESA模型。现在正在广泛使用的是MESA模型
在这里插入图片描述
管程中引入了条件变量的概念,而且每个条件变量都对应有一个等待队列。条件变量和等待队列的作用是解决线程之间的同步问题。

入口等待队列: 例A和B竞争锁(共享变量),B竞争失败,B就放入该队列中等待。

例子:AQS中所有等待获取锁资源的线程的队列。具体来说,当一个线程尝试获取锁资源失败时,它将会被加入到同步队列的末尾,并进入自旋等待状态,直到获取到锁资源或被唤醒。而当其他线程释放锁资源时,它们会从同步队列的头部选取一个或多个线程,并将其移动到锁资源的拥有者上,从而使其可以继续执行。

条件队列: 例Condition接口 调用了 await 的线程加入条件队列;signal唤醒后的线程最终还是加入到入口等待队列。

例子:AQS中,每个条件变量(Condition)都有一个等待队列,用来存放等待该条件的线程。当一个线程调用条件变量的await()方法时,它将会自动释放当前持有的锁,并加入到对应条件变量的等待队列中。而当其他线程调用条件变量的signal()或signalAll()方法时,它们会从对应条件变量的等待队列中选取一个或多个线程,并将其从等待队列中移动到同步队列(Sync Queue)中,以便重新竞争获取锁资源。

小结:同步队列主要解决了多个线程同时请求同一个锁资源时的竞争问题。条件等待队列主要解决了线程等待特定条件的问题,避免了线程不必要的忙等待和性能损失。
 

2.先根据代码分析ReentrantLock的实现

ReentrantLock lock = new ReentrantLock(false);
lock.lock() //加锁
=======这里写业务代码=========
lock.unlock() //解锁

从上面代码可以看到,我们开发人员只需要在 lock和 unlock中间写业务代码,就能实现线程安全的业务逻辑,那我们可以猜想到,两个方法之间肯定是有保证同一时间段仅一个线程进入的锁。而java自带的魔法类 UnSafe中提供了CAS来保证原子操作,所以代码变成了下面这样。

ReentrantLock lock = new ReentrantLock(false);
lock.lock() //加锁
	while(true){
		if(Cas加锁){
             break;//走到此说明加锁成功,跳出循环
		}
    }
=======获取到锁的线程在这里执行业务代码=========
lock.unlock() //解锁

上面已经保证了线程安全,但是也存在很大的问题,假如有1000个线程进来,只有T0拿到了锁,剩余999个线程一直在空转? Cpu不堪重负,因此可以通过一些方案来处理没抢到锁的线程,例如 yeild,sleep。

ReentrantLock lock = new ReentrantLock(false);
lock.lock() //加锁
	while(true){
		if(Cas加锁){
             break;//走到此说明加锁成功,跳出循环
		}
		//让出CPU使用权,一直让也不合适,999个线程互让没有意义,搞的跟活锁似的,还是只有T0在执行
		Thread.yeild()
		//睡多久?不好确定,万一T0瞬间就完成了
        Thread.sleep(1;
        //最终方案:让加锁失败的线程阻塞在这里 
        LockSupport.park();
    }
=======获取到锁的线程在这里执行业务代码=========
lock.unlock() //解锁

上面的代码看着好像没问题了,但没抢到锁的999个线程难道一直阻塞下去??占着茅坑不拉屎,所以需要有人去唤醒他,而java自带的工具 LockSupport有park,自然也有unPark方法。 LockSupport.park(要唤醒的线程)

ReentrantLock lock = new ReentrantLock(false);
lock.lock() //加锁
	while(true){
		if(Cas加锁){
             break;//走到此说明加锁成功,跳出循环
		}
		//让出CPU使用权,一直让也不合适,999个线程互让没有意义,还是只有T0在执行
		Thread.yeild()
		//睡多久?不好确定,万一T0瞬间就完成了
        Thread.sleep(1;

		//将要阻塞的线程保存起来,例如保存到 HashSet(不好处理公平问题) 或 Queue
		 HashSet.add(Thread)LikedQueued.put(Thread)

        //最终方案:让加锁失败的线程阻塞在这里 
        LockSupport.park();
    }
=======获取到锁的线程在这里执行业务代码=========
lock.unlock() //解锁
thread = HashSet.get() ; thread = LikedQueued.take()
LockSupport.unpark(thread)

小结:三板斧 【自旋,Cas,LockSupport】+存储容器

至此,我们对实现一把锁有了一定概念,接下来就深入源码来查看JDK8中ReentrantLock的实现。
 

3. 源码阅读

ReentrantLock的锁分为公平锁和非公平锁,下面简单介绍一下:
公平:排队打饭,新来的人得排在你的后面。
非公平:排队打饭,新来的人可能会插队,排在你前面。

以下面这个demo为例子,来进行源码的调试。

/**
 * 模拟抢票场景  3个线程抢2张票
 */
public class ReentrantLockDemo {
    private final ReentrantLock lock = new ReentrantLock();//默认非公平
    private static volatile int tickets = 2; // 总票数
    public void buyTicket() {
        lock.lock(); // 获取锁
        try {
            if (tickets > 0) { // 还有票    读
                try {
                    Thread.sleep(10); // 休眠10ms
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName() + "购买了第" + tickets-- + "张票"); //写
            } else {
                System.out.println("票已经卖完了," + Thread.currentThread().getName() + "抢票失败");
            }
        } finally {
            //zheli 唤醒
            lock.unlock(); // 释放锁
        }
    }

    public static void main(String[] args) {
        ReentrantLockDemo ticketSystem = new ReentrantLockDemo();
        Thread[] threads = new Thread[3];
        for (int i = 0; i < threads.length; i++) {
            // 抢票
            threads[i]  =  new Thread(ticketSystem::buyTicket, "线程" + i);
        }
        // 启动线程
        for (Thread thread : threads) {
            thread.start();
        }
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        System.out.println("剩余票数:" + tickets);
    }
}

进入lock.lock() 的实现,再进去 sync.lock()的实现
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

此时出现了三个核心方法,我们一个一个讲:
tryAcquire(尝试获取锁)
在这里插入图片描述
else if 中代码块说明:
判断当前线程是否等于拿到锁的那个线程, 如果是则将当前 state + acquires(例 state+1),表示这是第N次重入,这里可清晰看出可重入锁的判断依据。 statue 解锁时需要循环调用直到state为0,才表示锁已释放 。

简单来说,tryAcquire方法就是尝试获取锁,获取锁成功方法返回true,当前线程就可以去执行业务代码了。

addWaiter(构建节点并入队)
在这里插入图片描述
至此,入队后,无论模式是否公平,都需要按队列的顺序去排队了。

acquireQueued(CLH队列中尝试获取锁,获取不到的阻塞)

predecessor 什么时候获取前节点会为null
在这里插入图片描述
在这里插入图片描述
到此处得注意下,shouldParkAfterFailedAcquire方法外是一个循环,也就是说节点第一次进来时,大多都会走到设置前置节点的waitStatus由0变为-1的逻辑,当第二次循环时才会执行parkAndCheckInterrupt阻塞线程。
在这里插入图片描述
至此线程就被阻塞了,新的线程进来,流程也是大同小异不再赘述;当线程被中断或unpark唤醒时,会走到上图的 return Thread.interrupted(),继续在for中尝试是否满足抢锁的条件,不满足就阻塞,以此类推。

所以接下来就开始讲下解锁逻辑:

当执行完业务代码,他就会执行 unlock()方法释放锁。
在这里插入图片描述
在这里插入图片描述
从尝试释放锁的逻辑可看出重入锁lock了多少次,就要释放多少次。(即调用多少次unlock)
在这里插入图片描述
在这里插入图片描述
到这里好像流程就走完了,其实不然,上图只是唤醒了下个节点,但是与头结点还保持着链接!从上面的源码中也并没有看到 lock.unlock方法中将【被唤醒线程与前节点的关联关系解除】。是的,其实他是在唤醒后去做的,就在 acquireQueued方法中的 p.next = null; // help GC

和synchronized的一些区别:
1.sync存在锁升级的过程,通过对象头2bit的标记记录着锁的级别。
2.都是可重入锁,但是reentrantLock加了几次锁,就得手动释放几次锁,否则其他线程也拿不到锁。
3.reentrantLock是可以被interrupt信号和release方法中断唤醒的,sync全靠操作系统,不可控,对于我们开发并不方便。

注:今天讲的源码中并未使用到条件等待队列(Condition接口)

4.其他

自问自答:
waitStatus 类似我们业务的 state字段,只是一个标记,例如 = -1时,表示当前节点的下个节点允许被唤醒,只看reentrantLock的话没必要这个属性,但是AQS是很多工具类的基类,考虑的比较全。

唤醒阻塞线程的2种方式,一种是 lock.unLock,其实就是去调用 LockSupport.unpark(要唤醒的线程),
另一种就是interrupt信号唤醒,不过要区分 park方法
LockSupport.park 和 LockSupport.park(this) 区别:
一旦用 interrupt去唤醒 park,这个线程以后再也不会被 park住了。而park(this)就不会存在该问题。

为什么selfInterrupt方法打了一个中断标记?

public final void acquire(int arg) {
    if (!tryAcquire(arg) &&
        acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
        //上面都为true时,这里会打一个中断标记,因为 parkAndCheckInterrupt 被唤醒的其中一种方式,是 interrupt
        //且方法中判断完标记后,会清除, 这样我们使用这个工具的人(也就是程序员)不知道他是怎么被唤醒的,所以他这里又打了个标记。其实我感觉框架直接用 isInterrupt就够了,没必要重新去打标记
        selfInterrupt();
}

AQS 定义了5个队列中节点状态:
值为0,初始化状态,表示当前节点在sync队列中,等待着获取锁。
CANCELLED,值为1,表示当前的线程被取消;可能是异常,中断引起之类,需要被废弃结束
SIGNAL,值为-1,表示当前节点的后继节点包含的线程可以运行了,也就是当前可以unpark;
CONDITION,值为-2,表示当前节点在等待condition,也就是在condition队列中;
PROPAGATE,值为-3,表示当前场景下后续的acquireShared能够得以执行;

最后附一份方法解析图
在这里插入图片描述

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

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

相关文章

7.分区表和分桶表

1.创建分区表 create table dept_partition(deptno int,dname string,loc int ) partitioned by (dt string) // 分区字段(date) row format delimited fields terminated by \t; 2.增删改查操作 2.1 插入数据 1&#xff09;导入本地数据 -- 创建一个名字为dt2022-06-14的…

R语言 | 输入与输出

目录 一、认识文件夹 1.1 getwd()函数 1.2 setwd()函数 1.3 file.path()函数 1.4 dir()函数 1.5 list.files()函数 1.6 file.exists()函数 1.7 file.rename()函数 1.8 file.create()函数 1.9 file.copy()函数 ​1.10 file.remove()函数 二、数据输出&#xff1a;ca…

单片机c51中断 — 中断扫描法行列式键盘

项目文件 文件 关于项目的内容知识点可以见专栏单片机原理及应用 的第五章&#xff0c;中断 在第4章中已介绍过行列式键盘的工作原理&#xff0c;并编写了相应的键盘扫描程序。但应注意的是&#xff0c;在单片机应用系统中&#xff0c;键盘扫描只是 CPU 工作的内容之一。CPU …

一文理清 TiDB 与 MySQL 中的常用字符集及排序规则

1. 字符集&#xff08;character set&#xff09; 1.1. 字符集与编码规则 字符集&#xff08;character set&#xff09;即为众多字符的集合。字符集为每个字符分配一个唯一的 ID&#xff0c;称为 “Code Point&#xff08;码点&#xff09;”。编码规则是将 Code Point 转换…

商户查询的缓存——缓存击穿问题

缓存击穿问题也叫热点key问题&#xff0c;就是一个被高并发访问并且缓存重建业务比较复杂的key突然失效了&#xff0c;无数的请求访问会在瞬间给数据库带来巨大的冲击 常见的解决方案有两种&#xff1a; 互斥锁&#xff08;高并发时性能较差&#xff09; 逻辑过期 基于互斥锁方…

ASN.1-PKCS10-x509

在国际标准ITU-T X.690 《Information technology – ASN.1 encoding rules: Specification of Basic Encoding Rules (BER), Canonical Encoding Rules (CER) and Distinguished Encoding Rules (DER)》中定义了ASN.1编码规则。对于一般数据类型&#xff08;比如Integer、octe…

【软件工程】自动化测试保证卓越软件工程能力(2)

本次内容我们抽象一个待测试的目标软件产品&#xff0c;产品是基于web开发的。 自动化平台不是独立存在的&#xff0c;必然有一个目标待测试产品&#xff0c;用自动化测试来反映产品功能是否还是好的。 产品抽象v1 第一个版本&#xff0c;使用者&#xff08;USER&#xff09;发…

配置本地Angular环境并使用VsCode调试Angular前端项目

配置本地Angular环境并使用VsCode调试Angular前端项目 配置本地Angular环境部署Node.Js本地环境配置一下环境变量 使用vscode调试Angular安装vscode 配置本地Angular环境 部署Node.Js本地环境 1 从官网下载node.js, 本文为(v16.13.0) 下载地址: https://nodejs.org/dist/v16.…

windows server 2016报错无法打开所需文件install.wim

报错的前提条件: 1.下载原版镜像后,使用UltraISO制作U盘系统盘。 2.正常安装系统,到“安装程序正在启动界面”时弹出错误窗口,报错“Windows无法打开所需的文件 E:\Source\install.win。请确保安装所需的所有文件可用,并重新启动安装。错误代码:0x80070026”。 问题原因…

【MySQL学习】MySQL表的复合查询

文章目录 前言一、案例准备二、基本查询三、多表查询四、子查询4.1 单行子查询4.2 多行子查询4.3 多列子查询4.4 FROM子句中的子查询4.5 合并查询4.5.1 UNION4.5.2 UNION ALL 五、自连接六、内外连接6.1 内连接6.2 外连接6.2.1 左外连接6.2.2 右外连接 前言 对MySQL表的基本查…

大数据系列——Flink理论

概述 Flink是一个对有界和无界数据流进行有状态计算的分布式处理引擎和框架&#xff0c;既可以处理有界的批量数据集&#xff0c;也可以处理无界的实时流数据&#xff0c;为批处理和流处理提供了统一编程模型&#xff0c;其代码主要由 Java 实现&#xff0c;部分代码由 Scala实…

Java——Java选择题复习(1)(Java基础,进程,多线程,操作系统)

1. 下面关于程序编译说法正确的是&#xff08;&#xff09; A. java语言是编译型语言&#xff0c;会把java程序编译成二进制机器指令直接运行 B. java编译出来的目标文件与具体操作系统有关 C. java在运行时才进行翻译指令 D. java编译出来的目标文件&#xff0c;可以运行在任意…

房地产中介迎来重磅文件,但核心目标仍是专业化规范化发展

5月8日下午&#xff0c;住房和城乡建设部、市场监管总局联合刊登重磅文件《关于规范房地产经纪服务的意见》&#xff08;以下简称《意见》&#xff09;&#xff0c;因其涉及对经纪服务收费等具体问题的指导&#xff0c;文件引发市场重点关注。 不过&#xff0c;在系统性梳理文…

Redisson cannot use an unresolved DNS server address问题解决

概述 本文记录Mac IDEA开发&#xff0c;公司 远程办公时遇到的两个问题&#xff0c;记录一下。 问题 cannot use an unresolved DNS server address: [fe80::1%en0]:53 在家里&#xff0c;连上公司的VPN后&#xff0c;即可打开公司内网&#xff0c;远程办公。一切正常。某…

JVM学习(七):运行时数据区(精讲)

目录 一、运行时数据区概述 1.1 概述 1.2 运行时数据区中的GC和OOM 二、程序计数器&#xff08;PC寄存器&#xff0c;Program Counter Register&#xff09; 2.1 介绍 2.2 作用 2.3 特点 2.4 使用案例 2.5 常见面试题 三、虚拟机栈&#xff08;Java Virtual Machi…

前端008_类别模块_新增功能

类别模块_新增功能 1、需求分析2、新增窗口实现3、列表引用新增组件4、关闭弹出窗口5、校验表单数据6、提交表单数据6.1、Mock 添加新增模拟接口6.2、Api 调用接口6.3、测试新增功能1、需求分析 点击 新增 按钮后,对话框形式弹出新增窗口输入分类信息后,点击 确定 提交表单数…

IDEA小技巧-Git的回滚强推代码找回

标题IDEA小技巧-Git的回滚&&强推&&代码找回 本地未Commit 新增文件 delete 变更文件 rollback 第一种方式 第二种方式 切换默认变更列表 Commit未push undo commit 仅适用于最后一次的提交进行回滚 drop commit 回滚 revert commit revert commi…

Winform从入门到精通(42)——ToolStripContainer(史上最全)

该控件提供了一个上、下、左、右、中,一共五个面板 在界面添加toolStrip1,如下图: 在界面添加toolStrip2,如下图: 在界面添加toolStrip3,如下图: 在界面添加toolStrip4,如下图: 整体效果如下: 这里没有先将ToolStripContainer放到窗体上,这是因为经过测试发现…

FFmpeg 安装和使用

首先下载 ffmpeg&#xff1a; https://www.gyan.dev/ffmpeg/builds/ 点击下载 截至到 2023/5/9 FFmpeg的最新版是 6.0 版本的&#xff0c;如果想要下载之前版本的也可以&#xff0c;我这里就下载6.0版本的了 下载完之后解压缩&#xff0c;进入 bin 目录就是 FFmpeg 的主体…

shell脚本中数组的运用

shell脚本中数组的运用 一、数组的介绍1、概念2、定义方法3、数组包括的数据类型 二、关于数组的操作1、获取数组长度2、获取数组各个元素的下标3、获取数组列表或单个元素4、对数组切片输出5、对数组里的元素替换6、删除单个元素或数组7、数组追加元素8、向函数传入数组的值 三…