网站首页 > 技术教程 正文
抽象队列同步器(Abstract Queued Synchronizer),简称AQS,它是实现JUC并发包中独占锁和共享锁的基础组件。虽然开发者一般不会直接使用AQS,但是了解AQS底层原理对于并发程序设计还是很有帮助的。
AQS的双向队列
AQS维护了一个FIFO双向队列(CLH队列),其内部通过节点 head 和 tail 记录队首和队尾的队列元素,队列元素的类型为 Node。Node 中的 thread 变量用来存放进入AQS队列里面的线程,可以认为 Node 就是线程的抽象。
在AQS中,状态是由 state 属性来表示的,它是 volatile 类型的成员变量。对于AQS来说,线程同步的关键就是对状态值 state 进行操作。
状态值state的访问方式有三种:
- getState()
 - setState()
 - compareAndSetState()
 
其中 compareAndSetState() 的实现依赖于 Unsafe 类的 compareAndSwapInt() 方法。
独占与共享
AQS定义了两种同步方式:
1. 独占方式(Exclusive):只有一个线程能执行,如 ReentrantLock;
2. 共享方式(Share):多个线程可以同时执行,如Semaphore、CountDownLatch、ReadWriteLock,CyclicBarrier等。
使用独占方式获取的资源是与具体线程绑定的。如果一个线程获取到了资源,就会标记是这个线程获取到了(设置 state=1)。其他线程尝试操作 state 时,会发现当前该资源不是自己持有的(state 不为0),会在获取失败后被阻塞。
共享方式的资源与具体线程是不相关的,允许多个线程使用。当多个线程去请求资源时通过CAS方式竞争获取资源。
当一个线程获取到了资源后,另外一个线程再次去获取时如果当前资源还能满足它的需要(state >0),当前线程可使用CAS方式对 state 再次设置,从而获取同步状态。关于CAS操作的底层实现,可以参考《Java面试必考问题:CAS如何保证原子性?》。
依赖AQS的同步工具
JUC并发包中的很多同步工具都用到了AQS。
- ReentrantLock 使用AQS保存锁重复持有的次数。当一个线程获取锁时,ReentrantLock记录当前获得锁的线程标识,用于检测是否重复获取,以及错误线程试图解锁操作时异常情况的处理。
 - Semaphore 使用AQS同步状态来保存信号量的当前计数。tryRelease 会增加计数,acquireShared 会减少计数。
 - CountDownLatch 使用AQS同步状态来表示计数。计数为0时,所有的 Acquire 操作(CountDownLatch的 await 方法)才可以通过。
 - ReentrantReadWriteLock 使用AQS同步状态中的16位保存写锁持有的次数,剩下的16位用于保存读锁的持有次数。
 - ThreadPoolExecutor Worker 利用AQS同步状态实现对独占线程变量的设置(tryAcquire 和 tryRelease)。
 
参考资料:《Java并发编程之美》
我会持续更新关于物联网、云原生以及数字科技方面的文章,用简单的语言描述复杂的技术,也会偶尔发表一下对IT产业的看法,欢迎大家关注、转发和评论。
猜你喜欢
- 2024-10-30 读写锁,你难道不需要了解一下吗?
 - 2024-10-30 谷歌云故障14个小时,系“队列突变大量积压”引起
 - 2024-10-30 什么是AQS及其原理(aqs作用)
 - 2024-10-30 码仔漫画:怎么给女朋友讲明白线程池?
 - 2024-10-30 面试官:谈谈这4种磁盘IO调度算法--CFQ、NOOP、Deadline、AS
 - 2024-10-30 QT的信号槽机制简介(qt信号槽优缺点)
 - 2024-10-30 Java AQS(AbstractQueuedSynchronizer)详解
 - 2024-10-30 AQS是什么(AQS是什么药品)
 - 2024-10-30 详解磁盘IO调度算法--CFQ、NOOP、Deadline、AS
 - 2024-10-30 基于AbstractQueuedSynchronizer的并发类实现
 
欢迎 你 发表评论:
- 10-23Excel计算工龄和年份之差_excel算工龄的公式year
 - 10-23Excel YEARFRAC函数:时间的"年份比例尺"详解
 - 10-23最常用的10个Excel函数,中文解读,动图演示,易学易用
 - 10-23EXCEL中如何计算截止到今日(两个时间中)的时间
 - 10-2390%人不知道的Excel神技:DATEDIF 精准计算年龄,告别手动算错!
 - 10-23计算工龄及工龄工资(90%的人搞错了):DATE、DATEDIF组合应用
 - 10-23Excel中如何计算工作日天数?用这两个函数轻松计算,附新年日历
 - 10-23怎样快速提取单元格中的出生日期?用「Ctrl+E」批量搞定
 
- 最近发表
 - 
- Excel计算工龄和年份之差_excel算工龄的公式year
 - Excel YEARFRAC函数:时间的"年份比例尺"详解
 - 最常用的10个Excel函数,中文解读,动图演示,易学易用
 - EXCEL中如何计算截止到今日(两个时间中)的时间
 - 90%人不知道的Excel神技:DATEDIF 精准计算年龄,告别手动算错!
 - 计算工龄及工龄工资(90%的人搞错了):DATE、DATEDIF组合应用
 - Excel中如何计算工作日天数?用这两个函数轻松计算,附新年日历
 - 怎样快速提取单元格中的出生日期?用「Ctrl+E」批量搞定
 - Excel日期函数之DATEDIF函数_excel函数datedif在哪里
 - Excel函数-DATEDIF求司龄_exceldatedif函数计算年龄
 
 
- 标签列表
 - 
- 下划线是什么 (87)
 - 精美网站 (58)
 - qq登录界面 (90)
 - nginx 命令 (82)
 - nginx .http (73)
 - nginx lua (70)
 - nginx 重定向 (68)
 - Nginx超时 (65)
 - nginx 监控 (57)
 - odbc (59)
 - rar密码破解工具 (62)
 - annotation (71)
 - 红黑树 (57)
 - 智力题 (62)
 - php空间申请 (61)
 - 按键精灵 注册码 (69)
 - 软件测试报告 (59)
 - ntcreatefile (64)
 - 闪动文字 (56)
 - guid (66)
 - abap (63)
 - mpeg 2 (65)
 - column (63)
 - dreamweaver教程 (57)
 - excel行列转换 (56)
 
 

本文暂时没有评论,来添加一个吧(●'◡'●)