学习AQS

news/2024/10/24 1:26:59/

面试的时候被问到AQS会不会一脸懵逼呢?今天来学习一下AQS吧!

看看这个听起来高大上又难以回答的东西到底是个啥?

1.什么是AQS

JUC包中 的很多同步器如ReentrantLock、Semaphore、CountDownLatch等等都是有一些基础的共同的行为,比如:等待队列、条件队列、独占获取、共享获取等等,这些抽象行为就被封装成一个抽象类,就是AbstractQueuedSynchronizer(抽象队列同步器)。

一般是通过一个内部类Sync继承 AQS

将同步器所有调用都映射到Sync对应的方法

 看图,抽象类AbstractQueuedSynchronizer的实现类几乎都是各种我们常见的JUC工具类中的内部类——各种Sync。

AQS具有以下特性:独占/共享,公平/非公平,可重入,阻塞等待队列,允许中断

2.AQS核心结构

1.AQS内部维护属性 volatile int state

state表示资源的可用状态

state的三种访问方式:

getState() setState() compareAndSetState()

2.定义了两种资源访问方式:

Exclusive-独占,只有一个线程能执行,如ReentrantLock

Share-共享,多个线程可以同时执行,如Semaphore/CountDownLatch

3.AQS实现时主要实现以下几种方法:

isHeldExclusively():该线程是否正在独占资源。只有用到condition才需要去实现它。

tryAcquire(int):独占方式。尝试获取资源,成功则返回true,失败则返回false。

tryRelease(int):独占方式。尝试释放资源,成功则返回true,失败则返回false。

tryAcquireShared(int):共享方式。尝试获取资源。负数表示失败;0表示成功,但没有剩余可用资源;正数表 示成功,且有剩余资源。

tryReleaseShared(int):共享方式。尝试释放资源,如果释放后允许唤醒后续等待结点返回true,否则返回 false。

3.AQS定义两种队列:

同步等待队列: 主要用于维护获取锁失败时入队的线程。

条件等待队列: 调用await()的时候会释放锁,然后线程会加入到条件队列,调用signal()唤醒的时候会把条件队 列中的线程节点移动到同步等待队列中,等待再次获得锁。

        /*** Status field, taking on only the values:*   SIGNAL:     The successor of this node is (or will soon be)*               blocked (via park), so the current node must*               unpark its successor when it releases or*               cancels. To avoid races, acquire methods must*               first indicate they need a signal,*               then retry the atomic acquire, and then,*               on failure, block.*   CANCELLED:  This node is cancelled due to timeout or interrupt.*               Nodes never leave this state. In particular,*               a thread with cancelled node never again blocks.*   CONDITION:  This node is currently on a condition queue.*               It will not be used as a sync queue node*               until transferred, at which time the status*               will be set to 0. (Use of this value here has*               nothing to do with the other uses of the*               field, but simplifies mechanics.)*   PROPAGATE:  A releaseShared should be propagated to other*               nodes. This is set (for head node only) in*               doReleaseShared to ensure propagation*               continues, even if other operations have*               since intervened.*   0:          None of the above** The values are arranged numerically to simplify use.* Non-negative values mean that a node doesn't need to* signal. So, most code doesn't need to check for particular* values, just for sign.** The field is initialized to 0 for normal sync nodes, and* CONDITION for condition nodes.  It is modified using CAS* (or when possible, unconditional volatile writes).*/volatile int waitStatus;

AQS 定义了5个队列中节点状态waitStatus:

值为0,初始化状态,表示当前节点在sync队列中,等待着获取锁。

CANCELLED,值为1,表示当前的线程被取消;

SIGNAL,值为-1,表示当前节点的后继节点包含的线程需要运行,也就是unpark;

CONDITION,值为-2,表示当前节点在等待condition,也就是在condition队列中;

PROPAGATE,值为-3,表示当前场景下后续的acquireShared能够得以执行

同步等待队列:

AQS当中的同步等待队列也称CLH队列,CLH队列是Craig、Landin、Hagersten三人发明的一种

基于双向链表数据结构的队列,是FIFO先进先出线程等待队列,Java中的CLH队列是原CLH队列的一个变种,线程由原自旋机制改为阻塞机制。

当前线程如果获取同步状态失败时,AQS则会将当前线程已经等待状态等信息构造成一个节点(Node)并将其 加入到CLH同步队列,

同时会阻塞当前线程 当同步状态释放时,会把首节点唤醒(公平锁),使其再次尝试获取同步状态。

通过signal或signalAll将条件队列中的节点转移到同步队列。(由条件队列转化为同步队列)

条件等待队列

AQS中条件队列是使用单向列表保存的,用nextWaiter来连接:

调用await方法阻塞线程; 当前线程存在于同步队列的头结点,调用await方法进行阻塞(从同步队列转化到条件队列)

ReentrantLock是一种基于AQS框架的应用实现,是JDK中的一种线程并发访问的同步手段,它的 功能类似于synchronized是一种互斥锁,可以保证线程安全。

ps:MESA模型

管程:指的是管理共享变量以及对共享变量的操作过程,让他们支持并发。

互斥:同一时刻只允许一个线程访问共享资源;

同步:线程之间如何通信、协作。

在管程的发展史上,先后出现过三种不同的管程模型,分别是Hasen模型、Hoare模型和MESA模 型。现在正在广泛使用的是MESA模型。

 管程中引入了条件变量的概念,而且每个条件变量都对应有一个等待队列。条件变量和等待队列的 作用是解决线程之间的同步问题。

其实认真读完了以上的内容,依然感觉理解的不是很深入,需要进一步的跟源码。


http://www.ppmy.cn/news/581569.html

相关文章

​LeetCode解法汇总2485. 找出中枢整数

目录链接: 力扣编程题-解法汇总_分享记录-CSDN博客 GitHub同步刷题项目: https://github.com/September26/java-algorithms 原题链接:力扣 描述: 给你一个正整数 n ,找出满足下述条件的 中枢整数 x : 1…

【数据结构与算法C++实现】2、二分查找与简单递归

原视频为左程云的B站教学 文章目录 1 二分法1.1 在有序数组中查找特定元素1.2 在一个有序数组中查找>某个数的最左侧的位置1.3 在一个有序数组中查找<某个数最右侧位置1.4 局部最小值问题&#xff08;无序数组使用二分法的案例&#xff09; 2 简单的递归思想 1 二分法 …

ps平面设计作业_海报设计

没想到在CSDN上记录的第一篇博客是关于ps的海报设计&#xff0c;可能是真的太简单了吧…毕竟也是第一次尝试用ps来做海报&#xff0c;感觉最难的还是如何实现自己的构想&#xff0c;把脑子里的东西表达出来。素材…有跟着一些教程做&#xff0c;也有直接下载的&#xff0c;素材…

高端大气的艺术海报的ps教程

Step 1海报尺寸宽和高为768*1152像素&#xff0c;分辨率150像素/英寸&#xff0c;背景白色。 Step 2 确定了之后复制背景图层&#xff0c;双击图层进入图层样式&#xff0c;选择渐变叠加&#xff0c;颜色渐变绿色到蓝色&#xff0c;线性渐变。 Step 3 用椭圆工具绘制正圆填充颜…

Ps大尺寸图导出成小尺寸

本画布采用RGB&#xff0c;72分辨率&#xff0c;8位 演示步骤&#xff1a; 1.如图所示&#xff0c;一张47.7Mb的图片&#xff0c;要求导出为2Mb以内的图片 2.文件——导出——存储为Web所用格式&#xff08;快捷键&#xff1a;Ctrl—Alt—Shift S&#xff09; 3.所需要的…

PS按照图片大小调整画布大小

1、 2、 3、白色的地方就没了&#xff1b;

PS海报制作的常用方法

一、直接表现 纪录片海报《我在故宫修文物》&#xff0c;每张海报上的小小缺口&#xff0c;是整张海报的亮点&#xff0c;同时海报的画面又很精致大气&#xff0c;让人一见便觉得惊艳。 二、字体设计 材质字 2.创意字 三、图形设计 双重曝光、波普风、文字穿插、图形创意 双…

PS如何批量处理图片尺寸大小?免费哟

1、在photoshop中我们可以先处理一张&#xff0c;通过“动作”&#xff0c;来记录你的操作记录&#xff0c;接着其它图片也根据之前的步骤一张一张的去处理。 2、显然&#xff0c;PS这种处理方案很耗时间&#xff0c;而且门槛有点高 3、最近发现有一款免费的工具&#xff0c;可…