JUC之AQS

本文解析了AbstractQueuedSynchronizer在Java并发工具中的核心作用,介绍了如何通过FIFO队列和Node节点实现锁的分配,以及ReentrantLock公平与非公平锁的acquire方法和流程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

AQS抽象的队列同步器

public abstract class AbstractQueuedSynchronizer
    extends AbstractOwnableSynchronizer
    implements java.io.Serializable {

AbstractQueuedSynchronizer 是用来实现锁或者其他同步器组件的公共基础部分的抽象实现,是重量级基础框架及整个JUC体系的基石,主要用于解决锁分配给谁的问题。
整体就是一个抽象的FIFO队列来完成资源获取线程的排队工作,并通过一个int类变量表示持有锁的状态。
在这里插入图片描述
和AQS有关的锁
在这里插入图片描述
**锁,面向锁的使用者。**定义了程序员和锁交互的使用层API,隐藏了实现细节,你调用即可。
同步器,面向锁的实现者。 统一规范并简化了锁的实现,将其抽象出来屏蔽了同步状态管理,同步队列的管理和维护、阻塞线程队列和通知、唤醒机制等,是一切锁和同步组件实现的--------公共基础部分。
加锁会导致阻塞 有阻塞就需要排队,实现排队必然需要队列
在这里插入图片描述
在这里插入图片描述
源码实现
在这里插入图片描述
AQS使用一个volatile的int类型的成员变量来表示同步状态,通过内置的FIFO队列来完成资源获取的排队工作。将每条要去抢占资源的线程封装成一个Node节点来实现锁的分配,通过CAS完成对State值的修改。

在这里插入图片描述
在这里插入图片描述
节点的State状态,也就是Node节点的waitstatus的状态
在这里插入图片描述
在这里插入图片描述ReentrantLock的原理
在这里插入图片描述
公平与非公平的实现
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
从非公平锁看起
在这里插入图片描述
在这里插入图片描述
重点是acquire方法里面三个方法的使用

 public final void acquire(int arg) {
        if (!tryAcquire(arg) &&
            acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
            selfInterrupt();
    }

在这里插入图片描述
三个方法以及三大流向
在这里插入图片描述
在这里插入图片描述
动画演示

初始状态state=0
在这里插入图片描述
Thread A 占有后
在这里插入图片描述
首先进入队列后,构造一个哨兵节点
在这里插入图片描述
第二个线程B进来后,构造对应的节点B,并且尾指针指向该节点,变为尾节点
在这里插入图片描述
虚拟节点的下一个节点指向节点B,B节点入队完成
在这里插入图片描述
在这里插入图片描述

相关的代码逻辑
在这里插入图片描述
线程C 的节点C入队在这里插入图片描述

流程3,acquireQueued修改节点的前驱节点的waitstatus=-1,处于等待被唤醒的状态
在这里插入图片描述
线程A unlock释放锁后,state变为0,线程的占有变为null
在这里插入图片描述
unparkSuccessor 头节点唤醒后继节点,B节点占位成功,B节点的之前的前驱节点虚拟节点出队,B节点的thread设置为null,变为新的虚拟头节点
在这里插入图片描述
模拟节点的取消流程
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
tryacquire方法

公平锁
在这里插入图片描述
addWaiter 加入队列的方法
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
acquireQueued的核心方法
在这里插入图片描述
在这里插入图片描述

### Java JUC AQS 并发编程 抽象队列同步器 使用教程 源码解析 #### 什么是AQS? `AbstractQueuedSynchronizer`(简称AQS),作为Java并发包中的核心组件之一,提供了用于实现锁和其他同步器的基础框架。它不仅简化了锁和同步工具的创建过程,还提高了这些工具的工作效率[^2]。 #### 类图结构与工作原理 AQS的设计围绕着一个FIFO(先进先出)等待队列展开,该队列由多个节点组成,每个节点代表了一个正在等待获取资源的线程。每当有新的竞争者未能立即获得所需资源时,就会被构造成一个新的节点并加入到这个队列之中;而当现有持有者释放其持有的资源之后,则会从队头开始依次唤醒后续等待者去尝试占有资源[^5]。 #### 同步模式分类 为了适应不同场景下的需求,AQS支持两种主要类型的同步方式——独占式以及共享式: - **独占式**:一次只允许单个线程访问临界区,在这种情况下其他任何试图进入同一区域内的请求都将被迫挂起直到前序操作完成为止; - **共享式**:允许多个读取者同时存在而不互相干扰,只要不存在写入动作发生即可保持一致性和安全性[^3]。 #### 自定义同步器的关键接口 对于想要利用AQS来构建特定行为逻辑的新类型而言,开发者通常需要重载以下几个抽象方法以适配具体的应用环境: - `tryAcquire(int arg)` 和 `tryRelease(int arg)` - `tryAcquireShared(int arg)` 及 `tryReleaseShared(int arg)` - `isHeldExclusively()` 上述函数分别对应于独占/共享模式下对资源的操作控制流程,通过合理地覆盖它们可以轻松打造出满足业务特性要求的各种高级别同步原语[^1]。 ```java public class CustomSync extends AbstractQueuedSynchronizer { protected boolean tryAcquire(int acquires) { // 实现具体的独占式获取逻辑 return super.tryAcquire(acquires); } protected boolean tryRelease(int releases) { // 实现具体的独占式释放逻辑 return super.tryRelease(releases); } } ``` #### 队列管理机制详解 在实际运行过程中,AQS内部维护了一条双向链表形式的数据结构用来存储各个待处理的任务单元。每当新成员到来之时便会调用`enqueue()`方法将其追加至末端位置上形成完整的链条关系网状链接,并且借助CAS指令保证整个插入过程的安全可靠性质不受外界因素影响破坏[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值