AbstractQueuedSynchronizer是JUC包下的一个重要的类,JUC下的关于锁相关的类(如:ReentrantLock)等大部分是以此为基础实现的。那么我们就来分析一下AQS的原理。

1:通过以前的了解,我们先明白几个有用信息。

  1:实现基于FIFO(一个先进先出的队列)

  2:通过一个原子变量(atomic int) state来标识锁的状态(获取和释放)

  3:子类应该通过自定义改变原子变量的方法来代表锁的获取和释放

  4:底层是基于unsafe包的CAS操作,我们这里不做说明。

2:既然是一个队列,那么我们看一下这个队列的节点,部分代码如下

  1. static final class Node {
  2.  
  3. static final Node SHARED = new Node();
  4.  
  5. static final Node EXCLUSIVE = null;
  6. static final int CANCELLED = 1;
  7.  
  8. static final int SIGNAL = -1;
  9.  
  10. static final int CONDITION = -2;
  11.  
  12. static final int PROPAGATE = -3;
  13.  
  14. volatile int waitStatus;
  15.  
  16. volatile Node prev;
  17.  
  18. volatile Node next;
  19.  
  20. volatile Thread thread;
  21.  
  22. Node nextWaiter;
  23.  
  24. final boolean isShared() {
  25. return nextWaiter == SHARED;
  26. }
  27.  
  28. final Node predecessor() throws NullPointerException {
  29. Node p = prev;
  30. if (p == null)
  31. throw new NullPointerException();
  32. else
  33. return p;
  34. }
  35.  
  36. Node() { // Used to establish initial head or SHARED marker
  37. }
  38.  
  39. Node(Thread thread, Node mode) { // Used by addWaiter
  40. this.nextWaiter = mode;
  41. this.thread = thread;
  42. }
  43.  
  44. Node(Thread thread, int waitStatus) { // Used by Condition
  45. this.waitStatus = waitStatus;
  46. this.thread = thread;
  47. }
  48. }

  这个类比较简单,我们可以看出这个队列其实就是一个保存了线程信息的双向链表。其中 SHARED和EXCLUSIVE这两个属性分别标识了共享和独占(共享锁和独占锁)。我们特殊关注一下waitStatus这个属性,他有以下几个状态

  SIGNAL: 表明它的下一个节点的线程正在被阻塞(park);当前节点释放锁或者被取消时,它必须唤醒(unpark)下一个节点的线程;
  CANCELLED:该节点因为超时或者中断被取消,该状态的节点永远不会改变当前状态(会一直保持 CANCELLED 状态),同时该节点永远不会再被阻塞。
  CONDITION:该节点目前位于一个条件队列,在其状态改变之前他不会转移到同步队列中,并且当他转移到同步队列时它的状态会被设置为默认值。
  PROPAGATE:共享同步模式会无条件的传播给其它节点,当节点为头结点时在 doReleaseShared 方法中被设置为该状态来保证状态继续传播。
  0:非上述4中状态,有可能是刚获取signal,此时它的值是0,也有可能是新建的head节点

  如果上面的有些状态你看的云里雾里,不明所以的话不要紧,可以先有个大致印象。在后续的代码中看到这些状态时再结合这些解释看,就会清晰不少。

3下面是几个比较基础的方法,我们可以看下

  1. private Node enq(final Node node) {
  2. for (;;) {
  3. Node t = tail;
  4. if (t == null) { // Must initialize
  5. if (compareAndSetHead(new Node()))
  6. tail = head;
  7. } else {
  8. node.prev = t;
  9. if (compareAndSetTail(t, node)) {
  10. t.next = node;
  11. return t;
  12. }
  13. }
  14. }
  15. }

  这个方法就是往队列尾部添加节点,比较简单,我们不再多说。

  1. private Node addWaiter(Node mode) {
  2. Node node = new Node(Thread.currentThread(), mode);
  3. // Try the fast path of enq; backup to full enq on failure
  4. Node pred = tail;
  5. if (pred != null) {
  6. node.prev = pred;
  7. if (compareAndSetTail(pred, node)) {
  8. pred.next = node;
  9. return node;
  10. }
  11. }
  12. enq(node);
  13. return node;
  14. }

  这个方法是对enq方法的封装,也没啥好说的。但是我们能看到先快速添加到队列尾部,失败的话再通过enq循环尝试添加。

  1. private void unparkSuccessor(Node node) {
  2. /*
  3. * If status is negative (i.e., possibly needing signal) try
  4. * to clear in anticipation of signalling. It is OK if this
  5. * fails or if status is changed by waiting thread.
  6. */
  7. int ws = node.waitStatus;
  8. if (ws < 0)
  9. compareAndSetWaitStatus(node, ws, 0);
  10.  
  11. /*
  12. * Thread to unpark is held in successor, which is normally
  13. * just the next node. But if cancelled or apparently null,
  14. * traverse backwards from tail to find the actual
  15. * non-cancelled successor.
  16. */
  17. Node s = node.next;
  18. if (s == null || s.waitStatus > 0) {
  19. s = null;
  20. for (Node t = tail; t != null && t != node; t = t.prev)
  21. if (t.waitStatus <= 0)
  22. s = t;
  23. }
  24. if (s != null)
  25. LockSupport.unpark(s.thread);
  26. }

  这个方法的作用很明显了,就是唤醒当前节点的后续节点。在这之前会尝试更改锁标志状态,如果失败了也没关系,因为后置节点的线程会继续更改。但是,你有没有发现,在查找非空非取消状态的节点的时候竟然从后往前找,这感觉不太合理啊。从前往后找不是能更快的找到后置非空非取消状态的节点吗?我们记着这个问题继续看。

4:接下来我们看下独占模式下的获取锁的代码

  1. public final void acquire(int arg) {
  2. if (!tryAcquire(arg) &&
  3. acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
  4. selfInterrupt();
  5. }protected boolean tryAcquire(int arg) {
  6. throw new UnsupportedOperationException();
  7. }

  acquire和release分别对应了加锁和解锁。但是方法体中tryAcquire和tryRelease方法没有具体实现,因为不同的锁对公平,非公平,重入,不可重入的要求不同,所以这部分的自由度比较高,需要自己定制。

  我们看acquire方法:首先通过tryAcquire来获取锁,如果获取失败,则通过addWaiter将当前线程添加到队列尾部,然后通过acquireQueued来判断当前节点的线程是该阻塞呢还是不断尝试获取锁。看下面代码

  1. final boolean acquireQueued(final Node node, int arg) {
  2. boolean failed = true;
  3. try {
  4. boolean interrupted = false;
  5. for (;;) {
  6. final Node p = node.predecessor();
  7. if (p == head && tryAcquire(arg)) {
  8. setHead(node);
  9. p.next = null; // help GC
  10. failed = false;
  11. return interrupted;
  12. }
  13. if (shouldParkAfterFailedAcquire(p, node) &&
  14. parkAndCheckInterrupt())
  15. interrupted = true;
  16. }
  17. } finally {
  18. if (failed)
  19. cancelAcquire(node);
  20. }
  21. }
  22.  
  23. private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {
  24. int ws = pred.waitStatus;
  25. if (ws == Node.SIGNAL)
  26. /*
  27. * This node has already set status asking a release
  28. * to signal it, so it can safely park.
  29. */
  30. return true;
  31. if (ws > 0) {
  32. /*
  33. * Predecessor was cancelled. Skip over predecessors and
  34. * indicate retry.
  35. */
  36. do {
  37. node.prev = pred = pred.prev;
  38. } while (pred.waitStatus > 0);
  39. pred.next = node;
  40. } else {
  41. /*
  42. * waitStatus must be 0 or PROPAGATE. Indicate that we
  43. * need a signal, but don't park yet. Caller will need to
  44. * retry to make sure it cannot acquire before parking.
  45. */
  46. compareAndSetWaitStatus(pred, ws, Node.SIGNAL);
  47. }
  48. return false;
  49. }
  50.  
  51. private final boolean parkAndCheckInterrupt() {
  52. LockSupport.park(this);
  53. return Thread.interrupted();
  54. }

  acquireQueued方法会判断当前节点的前置节点是不是head节点,如果是且尝试获取到了锁(说明head节点已经释放锁),那么则设置当前节点为head节点,当先线程也不需要阻塞。如果前置阶段不为head节点或者尝试获取锁失败,那么就通过shouldParkAfterFailedAcquire方法来判断该线程是不是应该阻塞。

  在shouldParkAfterFailedAcquire方法中我们可以看到对各种 waitStatus 状态的处理。特别注意ws>0时的处理:这段逻辑将队列中最后一个节点链接到了前一个没有CANCELLED的节点,即剔除了中间状态为CANCELLED的节点。这个确保了最后节点的前一个节点的状态为SIGNAL。这样的话下次循环该线程就可以放心park了。

5:接下来我们看下独占模式下的释放锁的代码

  1. public final boolean release(int arg) {
  2. if (tryRelease(arg)) {
  3. Node h = head;
  4. if (h != null && h.waitStatus != 0)
  5. unparkSuccessor(h);
  6. return true;
  7. }
  8. return false;
  9. }
  10.  
  11. protected boolean tryRelease(int arg) {
  12. throw new UnsupportedOperationException();
  13. }

  代码逻辑比较清晰:当锁释放成功后,要唤醒下一个节点的线程。同样的tryRelease需要自己实现。

  我们看看是如何唤醒下个节点的线程的。

  1. private void unparkSuccessor(Node node) {
  2. /*
  3. * If status is negative (i.e., possibly needing signal) try
  4. * to clear in anticipation of signalling. It is OK if this
  5. * fails or if status is changed by waiting thread.
  6. */
  7. int ws = node.waitStatus;
  8. if (ws < 0)
  9. compareAndSetWaitStatus(node, ws, 0);
  10.  
  11. /*
  12. * Thread to unpark is held in successor, which is normally
  13. * just the next node. But if cancelled or apparently null,
  14. * traverse backwards from tail to find the actual
  15. * non-cancelled successor.
  16. */
  17. Node s = node.next;
  18. if (s == null || s.waitStatus > 0) {
  19. s = null;
  20. for (Node t = tail; t != null && t != node; t = t.prev)
  21. if (t.waitStatus <= 0)
  22. s = t;
  23. }
  24. if (s != null)
  25. LockSupport.unpark(s.thread);
  26. }

  这个逻辑也比较简单,从后往前找到第一个状态不为CANCELLED的节点,并通过unpark唤醒它的线程。(注意这里仍然是从后向前遍历)

  这里要注意一点:当head和s之间存在CANCELLED节点时,s.prev节点是CANCELLED节点(这点可能会在6里出问题)

6:线程被唤醒后? 

  1. final boolean acquireQueued(final Node node, int arg) {
  2. boolean failed = true;
  3. try {
  4. boolean interrupted = false;
  5. for (;;) {
  6. final Node p = node.predecessor();
  7. if (p == head && tryAcquire(arg)) {
  8. setHead(node);
  9. p.next = null; // help GC
  10. failed = false;
  11. return interrupted;
  12. }
  13. if (shouldParkAfterFailedAcquire(p, node) &&
  14. parkAndCheckInterrupt())
  15. interrupted = true;
  16. }
  17. } finally {
  18. if (failed)
  19. cancelAcquire(node);
  20. }
  21. }

  前面我们看到线程是在 acquireQueued 方法的 parkAndCheckInterrupt() 这被阻塞的。当它被唤醒后,仍在这个循环里。下次循环就能获取到锁了。

  但是被唤醒节点的前置节点一定是head节点吗?理论上是的,但是通过5我们知道也可能不是head节点,而是一个CANCELLED节点。

  那么被唤醒节点的前置节点是CANCELLED节点怎么处理了呢。根据逻辑,又进入了  shouldParkAfterFailedAcquire 方法,这个方法会清除两个节点间的CANCELLED节点。在经过这个方法后,就能保证在下次循环中被唤醒节点的前置节点就是head节点。

6:为什么从后向前遍历?

  看了半天也没看出来为啥从后向前遍历。

  我们看了 volatile Node next  属性的注释

  1. /**
  2. * Link to the successor node that the current node/thread
  3. * unparks upon release. Assigned during enqueuing, adjusted
  4. * when bypassing cancelled predecessors, and nulled out (for
  5. * sake of GC) when dequeued. The enq operation does not
  6. * assign next field of a predecessor until after attachment,
  7. * so seeing a null next field does not necessarily mean that
  8. * node is at end of queue. However, if a next field appears
  9. * to be null, we can scan prev's from the tail to
  10. * double-check. The next field of cancelled nodes is set to
  11. * point to the node itself instead of null, to make life
  12. * easier for isOnSyncQueue.
  13. */
  14. volatile Node next;

  红字表出来的意思是,"enq操作在当前节点加入队列后,才将前置节点的next指向最后的节点。这说明我们从前向后遍历时,看到一个next为null的节点并不意味着他是最后一个节点。但是从后向前遍历却能避免这个问题"

  结合代码我们再看一下

  1. private Node enq(final Node node) {
  2. for (;;) {
  3. Node t = tail;
  4. if (t == null) { // Must initialize
  5. if (compareAndSetHead(new Node()))
  6. tail = head;
  7. } else {
  8. 8 node.prev = t;
  9. 9 if (compareAndSetTail(t, node)) {
  10. 10 t.next = node;
  11. return t;
  12. }
  13. }
  14. }
  15. }

  对应的8,9,10行。因为不是原子操作,有可能出现:第9行执行成功后(此时新节点已经添加到了队列尾部),有个线程从前到后遍历各个节点,由于前置节点t.next==null,所以新追加到队列尾部的节点无法被扫描到。相反的从后向前的话,第8行就避免了这个问题。

7:总结

  1:AQS的本质是CAS自旋 volatile 变量

  2:阻塞的线程被有序的排列在FIFO中

  3:线程的阻塞和唤醒用的是LockSupport.park()和LockSupport.unpark()

  4: sleep, wait, park的区别:

    sleep, 进入TIMED_WAITING状态,不出让锁;

    wait, 进入TIMED_WAITING状态,出让锁,并进入对象的等待队列,必须结合sychronized使用。

    park, 进入WAITING状态,对比wait不需要获得锁就可以让线程WAITING,通过unpark唤醒

关于线程阻塞和等待状态的区别见:https://blog.csdn.net/Mrxingyong/article/details/95164329

  

AQS(AbstractQueuedSynchronizer)解析的更多相关文章

  1. 高并发第十一弹:J.U.C -AQS(AbstractQueuedSynchronizer) 组件:Lock,ReentrantLock,ReentrantReadWriteLock,StampedLock

    既然说到J.U.C 的AQS(AbstractQueuedSynchronizer)   不说 Lock 是不可能的.不过实话来说,一般 JKD8 以后我一般都不用Lock了.毕竟sychronize ...

  2. AQS(AbstractQueuedSynchronizer)介绍-01

    1.概述 AQS( AbstractQueuedSynchronizer ) 是一个用于构建锁和同步器的框架,许多同步器都可以通过AQS很容易并且高效地构造出来.如: ReentrantLock 和 ...

  3. 从ReentrantLock看AQS (AbstractQueuedSynchronizer) 运行流程

    从ReentrantLock看AQS (AbstractQueuedSynchronizer) 运行流程 概述 本文将以ReentrantLock为例来讲解AbstractQueuedSynchron ...

  4. AQS原理解析 AbstractQueuedSynchronizer

    AQS实现原理  https://blog.csdn.net/ym123456677/article/details/80381354   https://www.cnblogs.com/keleli ...

  5. 高并发第十单:J.U.C AQS(AbstractQueuedSynchronizer) 组件:CountDownLatch. CyclicBarrier .Semaphore

    这里有一篇介绍AQS的文章 非常好: Java并发之AQS详解 AQS全名:AbstractQueuedSynchronizer,是并发容器J.U.C(java.lang.concurrent)下lo ...

  6. 高并发编程-AQS深入解析

    要点解说 AbstractQueuedSynchronizer简称AQS,它是java.util.concurrent包下CountDownLatch/FutureTask/ReentrantLock ...

  7. AbstractQueuedSynchronizer解析

    AbstractQueuedSynchronizer简称为AQS,是juc里很基本的一个包,juc里很多工具类是基于AQS实现的,理解了AQS,其它很多juc工具类也会比较清楚了. 1.方法简述 ge ...

  8. 5. AQS(AbstractQueuedSynchronizer)抽象的队列式的同步器

    5.1 AbstractQueuedSynchronizer里面的设计模式--模板模式 模板模式:父类定义好了算法的框架,第一步做什么第二步做什么,同时把某些步骤的实现延迟到子类去实现. 5.1.1 ...

  9. AQS(AbstractQueuedSynchronizer)应用案例-02

    1.概述 通过对AQS源码的熟悉,我们可以通过实现AQS实现自定义的锁来加深认识. 2.实现 1.首先我们确定目标是实现一个独占模式的锁,当其中一个线程获得资源时,其他线程再来请求,让它进入队列进行公 ...

随机推荐

  1. Python使用QQ邮箱发送邮件报错smtplib.SMTPAuthenticationError

    最新在学习Python的基础入门系列课程,今天学习到使用python 的内置库smtplib发送邮件内容. 使用Python发送邮件步骤简单: 创建SMTP连接 使用邮箱和密码登录SMTP服务器 创建 ...

  2. 《MySQL数据库》常用语法(二)

    表关联关系: -- 内联接 SELECT * FROM m INNER JOIN n ON m.id = n.id; -- 左外联接 SELECT * FROM m LEFT JOIN n ON m. ...

  3. 安装完PyCharm,启动时弹出Failed to load JVM DLLbinserverjvm

    安装完PyCharm,启动时弹出"Failed to load JVM DLL\bin\server\jvm.dll"解决方案 问题描述:打开PyCharm时,弹出"Fa ...

  4. AWVS 10.5使用指南

    前言 AWVS是一款可与IBM AppScan比肩的.功能十分强大的Web漏洞扫描器.由Acunetix开发,官方站点提供了关于各种类型漏洞的解释和如何防范,具体参考:Acunetix Web Vul ...

  5. HttpRunner学习4--使用正则表达式提取数据

    前言 在HttpRunner中,我们可通过extract提取数据,当响应结果为 JSON 结构,可使用 content 结合 . 运算符的方式,如 content.code,用起来十分方便,但如果响应 ...

  6. sqoop 安装与使用

    Sqoop(发音:skup)是一款开源的工具,主要用于在Hadoop(Hive)与传统的数据库间进行数据的传递,可以将一个关系型数据库(例如 : MySQL ,Oracle ,Postgres等)中的 ...

  7. linux(center OS7)安装JDK、tomcat、mysql 搭建java web项目运行环境

    一.安装JDK 1.卸载旧版本或者系统自带的JDK (1)列出所有已安装的JDK rpm -qa | grep jdk (2)卸载不需要的JDK yum -y remove 安装包名称 2.下载并解压 ...

  8. Mysql - 高可用方案之MMM(二)

    一.概述 上一篇博客中(https://www.cnblogs.com/ddzj01/p/11535796.html)介绍了如何搭建MMM架构,本文将通过实验介绍MMM架构的优缺点. 二.优点 1. ...

  9. Spring Boot 为什么这么火?

    没错 Spring Boot 越来越火了,而且火的超过了我的预期,作为一名行走一线的 Java 程序员,你可能在各个方面感受到了 Spring Boot 的火. Spring Boot 的火 技术社区 ...

  10. 对python函数后面有多个括号的理解?

    一般而言,函数后面只有一个括号.如果看见括号后还有一个括号,说明第一个函数返回了一个函数,如果后面还有括号,说明前面那个也返回了一个函数.以此类推. 比如fun()() def fun(): prin ...