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 并不是将每个方法都在同一个锁上同步 ...
 
随机推荐
- CSS劫持攻击
			
CSS劫持攻击 CSS劫持是一种并不很受重视的劫持方式,但是其也有一定的危害,且由于其并不一定需要依赖JavaScript,这使得此种攻击方式更容易实现. ClickJacking点击劫持 当访问某网 ...
 - dijkstra模板题 洛谷1339 邻接图建边
			
题目链接:https://www.luogu.com.cn/problem/P1339 朴素dijkstra算法的复杂度是O(n^2),用堆优化的dijkstra复杂度是O(nlogn)的.在本题中前 ...
 - fastText 训练和使用
			
2019-09-09 16:33:11 问题描述:fastText是如何进行文本分类的. 问题求解: fastText是一种Facebook AI Research在16年开源的一个文本分类器. 其特 ...
 - OpenCV-Python 理解K近邻 | 五十三
			
目标 在本章中,我们将了解k近邻(kNN)算法的原理. 理论 kNN是可用于监督学习的最简单的分类算法之一.这个想法是在特征空间中搜索测试数据的最近邻.我们将用下面的图片来研究它. 在图像中,有两个族 ...
 - Redis在linux环境下的安装
			
下载Redis安装包 wget http://download.redis.io/releases/redis-3.2.9.tar.gz 解压Redis安装包 tar -zxvf redis-3.2. ...
 - iOS 继承
			
是否使用继承需要考虑三个点: 父类只是给子类提供服务,并不涉及子类的业务逻辑 层级关系明显,功能划分清晰,父类和子类各做各的. 父类的所有变化,都需要在子类中体现,也就是说此时耦合已经成为需求 万不得 ...
 - 添加属于自己的python模块空间
			
在我们学习python的过程中会遇到很多时候,我们需要自己曾经写过的模块,它可能是一个函数或者其他的东西,,,, 下面是我的解决过程,如果你像将自己建立的文件夹当作你存放自己写的模块的地方,你需要将你 ...
 - win10 安装redis相关问题。
			
最近需要在win10安装redis,但是redis的msi文件总是报这个错误: Redis on Windows Setup Wizard ended prematurely 都说是.NET fram ...
 - html中的字幕滚动marquee属性
			
该标签不是HTML3.2的一部分,并且只支持MSIE3以后内核,所以如果你使用非IE内核浏览器(如:Netscape)可能无法看到下面一些很有意思的效果,该标签是个容器标签. 语法: <marq ...
 - Django REST Framework 教程开篇
			
作者:HelloGitHub-追梦人物 欢迎来到 HelloDjango 全栈系列教程第二步--Django REST Framework 教程! 首先恭喜你完成了 HelloDjango 全栈系列教 ...