JUC并发编程基石AQS源码之结构篇
前言
AQS(AbstractQueuedSynchronizer)算是JUC包中最重要的一个类了,如果你想了解JUC提供的并发编程工具类的代码逻辑,这个类绝对是你绕不过的。我相信如果你是第一次看AQS源码肯定是一脸懵逼,一个个方法跳来跳去一会就绕蒙了。所以把整个代码骨架搞明白是你看懂AQS源码的第一步。本篇文章只说代码结构,之后的篇章会讲解AQS具体的执行逻辑。
顶级接口Lock
public interface Lock {
void lock();
void unlock();
void lockInterruptibly() throws InterruptedException;
boolean tryLock();
boolean tryLock(long time, TimeUnit unit) throws InterruptedException;
Condition newCondition();
}
我们一直强调面向接口编程的思想,看源码也是先从接口入手。Lock算是JUC提供的锁中的顶级接口,我们平常使用的ReentrantLock、ReadWriteLock等直接或间接的都实现了这个接口。这个接口主要是用来规定我们怎样去加锁和解锁,将加锁和解锁的方法暴露出去。下面的伪代码是一个典型的加锁解锁方式,多么简单,这就是面向接口的艺术。
Lock l = ...;
l.lock();
try {
// 自己的逻辑代码
} finally {
l.unlock();
}
代码结构
到现在我们知道了怎么加锁和解锁的方式,下一步自己通过接口去实现一个加锁类。这个比你直接看源码更重要。Doug Lea(AQS源码作者)大神在AQS类注解上就给我们提供了一个示例类Mutex。看源码类注解和方法注解也是你理解源码的一个重要渠道,还可以锻炼自己的英文。
实现Lock接口
首先要实现Lock接口,将加锁和解锁方法暴露出去。我们以加锁为例
class Mutex implements Lock, java.io.Serializable {
public void lock() {
sync.acquire(1);
}
代码很简单,调用了sync.acquire(1)方法,所以这个方法是加锁逻辑的入口,往下看
内部类继承AQS
现在将加锁方法暴露出去了,具体的加锁逻辑则需要AQS类了。AQS是一个抽象类,我们要使用它则需要继承,实现抽象方法。AQS加锁采用的是模板的设计模式,加锁以及锁失败的后续处理的整体流程代码已经实现,我们只需要实现我们需要的具体加锁方式即可。
class Mutex implements Lock, java.io.Serializable {
// The sync object does all the hard work. We just forward to it.
private final Sync sync = new Sync();
// 内部类继承AQS
private static class Sync extends AbstractQueuedSynchronizer {
// 实现抽象方法
public boolean tryAcquire(int acquires) {
assert acquires == 1; // Otherwise unused
if (compareAndSetState(0, 1)) {
setExclusiveOwnerThread(Thread.currentThread());
return true;
}
return false;
}
}
}
我们会发现Sync继承了AQS,但是没有sync.acquire()这个方法,那么这个方法肯定来源于父类了。
AQS的acquire方法
public abstract class AbstractQueuedSynchronizer{
//加锁的入口方法,模板的设计模式
public final void acquire(int arg) {
if (!tryAcquire(arg) &&
acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
selfInterrupt();
}
//具体的加锁逻辑,需要自己实现
protected boolean tryAcquire(int arg) {
throw new UnsupportedOperationException();
}
acquire方法定义了整个的加锁流程,而且使用了设计模式中的模板模式。
整体调用流程

从上面的流程可以看出,就这四个步骤就涉及到了三个类
Mutex.lock为暴露出去的加锁方法
AQS.acquire是加锁的模板方法,实现了加锁逻辑的整个流程
Sync.tryAcquire,图中的绿色部分,是一个抽象方法需要自己实现,针对不同的锁类型如公平锁、非公平锁、共享锁、独占锁等有不同的实现方式。
AQS.acquireQueued是加锁失败后的逻辑,将线程入队,这个后面讲AQS源码会重点讲。
解锁操作的流程和加锁类似,读者可以自己看一下解锁的流程。
自定义加锁类的源码
下面的代码是Mutex类的整体代码,有需要的可以在自己的IDE中感受一下整体的结构。
class Mutex implements Lock, java.io.Serializable {
public static void main(String[] args) {
Lock lock = new Mutex();
lock.lock();
try {
}finally {
lock.unlock();
}
}
public void lock() {
sync.acquire(1);
}
public void unlock() {
sync.release(1);
}
// Our internal helper class
private static class Sync extends AbstractQueuedSynchronizer {
// Reports whether in locked state
protected boolean isHeldExclusively() {
return getState() == 1;
}
// Acquires the lock if state is zero
public boolean tryAcquire(int acquires) {
assert acquires == 1; // Otherwise unused
if (compareAndSetState(0, 1)) {
setExclusiveOwnerThread(Thread.currentThread());
return true;
}
return false;
}
// Releases the lock by setting state to zero
protected boolean tryRelease(int releases) {
assert releases == 1; // Otherwise unused
if (getState() == 0) throw new IllegalMonitorStateException();
setExclusiveOwnerThread(null);
setState(0);
return true;
}
// Provides a Condition
Condition newCondition() {
return new ConditionObject();
}
// Deserializes properly
private void readObject(ObjectInputStream s)
throws IOException, ClassNotFoundException {
s.defaultReadObject();
setState(0); // reset to unlocked state
}
}
// The sync object does all the hard work. We just forward to it.
private final Sync sync = new Sync();
public boolean tryLock() {
return sync.tryAcquire(1);
}
public Condition newCondition() {
return sync.newCondition();
}
public boolean isLocked() {
return sync.isHeldExclusively();
}
public boolean hasQueuedThreads() {
return sync.hasQueuedThreads();
}
public void lockInterruptibly() throws InterruptedException {
sync.acquireInterruptibly(1);
}
public boolean tryLock(long timeout, TimeUnit unit)
throws InterruptedException {
return sync.tryAcquireNanos(1, unit.toNanos(timeout));
}
}
下一篇我们将根据上面所说的来分析ReentrantLock类的代码结构
如有不实,还望指正
JUC并发编程基石AQS源码之结构篇的更多相关文章
- JUC 并发编程--11, AQS源码原理解析, ReentrantLock 源码解读
这里引用别人博客,不重复造轮子 https://blog.csdn.net/u012881584/article/details/105886486 https://www.cnblogs.com/w ...
- JUC并发编程基石AQS之主流程源码解析
前言 由于AQS的源码太过凝练,而且有很多分支比如取消排队.等待条件等,如果把所有的分支在一篇文章的写完可能会看懵,所以这篇文章主要是从正常流程先走一遍,重点不在取消排队等分支,之后会专门写一篇取消排 ...
- 多线程进阶——JUC并发编程之CountDownLatch源码一探究竟
1.学习切入点 JDK的并发包中提供了几个非常有用的并发工具类. CountDownLatch. CyclicBarrier和 Semaphore工具类提供了一种并发流程控制的手段.本文将介绍Coun ...
- 多线程高并发编程(3) -- ReentrantLock源码分析AQS
背景: AbstractQueuedSynchronizer(AQS) public abstract class AbstractQueuedSynchronizer extends Abstrac ...
- JUC锁:核心类AQS源码详解
目录 1 疑点todo和解疑 2 AbstractQueuedSynchronizer学习总结 2.1 AQS要点总结 2.2 细节分析 2.2.1 插入节点时先更新prev再更新前驱next 2.2 ...
- 并发编程之 AQS 源码剖析
前言 JDK 1.5 的 java.util.concurrent.locks 包中都是锁,其中有一个抽象类 AbstractQueuedSynchronizer (抽象队列同步器),也就是 AQS, ...
- 多线程高并发编程(10) -- ConcurrentHashMap源码分析
一.背景 前文讲了HashMap的源码分析,从中可以看到下面的问题: HashMap的put/remove方法不是线程安全的,如果在多线程并发环境下,使用synchronized进行加锁,会导致效率低 ...
- 多线程高并发编程(7) -- Future源码分析
一.概念 A Future计算的结果. 提供方法来检查计算是否完成,等待其完成,并检索计算结果. 结果只能在计算完成后使用方法get进行检索,如有必要,阻塞,直到准备就绪. 取消由cancel方法执行 ...
- 并发编程实战-ConcurrentHashMap源码解析
jdk8之前的实现原理 jdk1.7中采用的数据结构是Segment + HashEntry 的方式进行实现.主要的结构如下图: ConcurrentHashMap 并不是将每个方法都在同一个锁上同步 ...
随机推荐
- jQuery的简单用法(jQuery的简介,选择器,属性和css,文档处理)
一.jQuery简介 1.1. JS库 JavaScript 库封装了很多预定义的对象和实用函数.能帮助使用者建立有高难度交互客户端页面, 并且兼容各大浏览器. 1.2. 当前流行的 JavaSc ...
- vue 指令,成员,组件
目录 复习 v-once指令 v-cloak指令(了解) 条件指令 v-pre指令(了解) 循环指令 todolist留言板案例 实例成员 - 插值表达式符号(了解) 计算属性 属性监听 组件 局部组 ...
- python爬取网站页面时,部分标签无指定属性而报错
在写爬取页面a标签下href属性的时候,有这样一个问题,如果a标签下没有href这个属性则会报错,如下: 百度了有师傅用正则匹配的,方法感觉都不怎么好,查了BeautifulSoup的官方文档,发现一 ...
- 痞子衡嵌入式:测一测i.MXRT1170 Raw NAND启动时间(从POR到进App的Reset_Handler)
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是恩智浦i.MX RT1170 Raw NAND启动时间. 关于i.MXRT1170这颗划时代的MCU,痞子衡去年10月在其刚发布的时候, ...
- UNIX环境高级编程——TCP/IP网络编程 常用网络信息检索函数
UNIX环境高级编程——TCP/IP网络编程 常用网络信息检索函数 gethostname() getppername() getsockname() gethostbyname() ...
- matplotlib.pyplot库函数关于坐标轴显示的支持
matplotlib.pyplot库函数关于坐标轴显示的支持 https://blog.csdn.net/Stark_595/article/details/80787005?depth_1-utm_ ...
- OpenCV-Python OpenCV中的K-Means聚类 | 五十八
目标 了解如何在OpenCV中使用cv.kmeans()函数进行数据聚类 理解参数 输入参数 sample:它应该是np.float32数据类型,并且每个功能都应该放在单个列中. nclusters( ...
- Selenium系列(十五) - Web UI 自动化基础实战(2)
如果你还想从头学起Selenium,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1680176.html 其次,如果你不懂前端基础知识, ...
- coding++:Spring中的@Transactional(rollbackFor = Exception.class)属性详解
异常: 如下图所示,我们都知道 Exception 分为 运行时异常 RuntimeException 和 非运行时异常. error 是一定会回滚的. 如果不对运行时异常进行处理,那么出现运行时异常 ...
- Python终端打印彩色文字
终端彩色文字 class Color_f: black = 30 red = 31 green = 32 yellow= 33 blue = 34 fuchsia=35 cyan = 36 white ...