网站首页 > 技术教程 正文
抽象队列同步器(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的并发类实现
你 发表评论:
欢迎- 最近发表
-
- linux CentOS检查见后门程序的shell
- 网络安全工程师演示:黑客是如何使用Nmap网络扫描工具的?
- Linux中ftp服务修改默认21端口等(linux修改ftp配置文件)
- Linux系统下使用Iptables配置端口转发,运维实战收藏!
- 谈谈TCP和UDP源端口的确定(tcp和udp的端口号相同吗)
- Linux 系统 通过端口号找到对应的服务及相应安装位置
- 快速查找NAS未占用端口!Docker端口秒级排查+可视化占坑双杀技
- 【知识杂谈#2】如何查看Linux的(本地与公网)IP地址与SSH端口号
- 如何在Linux中查询 DNS 记录,这三个命令可谓是最常用、最经典的
- 【Linux系统编程】特殊进程之守护进程
- 标签列表
-
- 下划线是什么 (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)
本文暂时没有评论,来添加一个吧(●'◡'●)