每一个Java工程师应该都或多或少了解过AQS,我自己也是前前后后,反反复复研究了很久,看了忘,忘了再看,每次都有不一样的体会。这次趁着写博客,打算重新拿出来系统的研究下它的源码,总结成文章,便于以后复习。

原文地址:http://www.jianshu.com/p/71449a7d01af

AbstractQueuedSynchronizer(以下简称AQS)作为java.util.concurrent包的基础,它提供了一套完整的同步编程框架,开发人员只需要实现其中几个简单的方法就能自由的使用诸如独占,共享,条件队列等多种同步模式。我们常用的比如ReentrantLock,CountDownLatch等等基础类库都是基于AQS实现的,足以说明这套框架的强大之处。鉴于此,我们开发人员更应该了解它的实现原理,这样才能在使用过程中得心应手。

总体来说个人感觉AQS的代码非常难懂,本文就其中的独占锁实现原理进行分析。

一、执行过程概述

首先先从整体流程入手,了解下AQS独占锁的执行逻辑,然后再一步一步深入分析源码。

获取锁的过程:

  1. 当线程调用acquire()申请获取锁资源,如果成功,则进入临界区。
  2. 当获取锁失败时,则进入一个FIFO等待队列,然后被挂起等待唤醒。
  3. 当队列中的等待线程被唤醒以后就重新尝试获取锁资源,如果成功则进入临界区,否则继续挂起等待。

释放锁过程:

  1. 当线程调用release()进行锁资源释放时,如果没有其他线程在等待锁资源,则释放完成。
  2. 如果队列中有其他等待锁资源的线程需要唤醒,则唤醒队列中的第一个等待节点(先入先出)。

二、源码深入分析

基于上面所讲的独占锁获取释放的大致过程,我们再来看下源码实现逻辑:

首先来看下获取锁的方法acquire()

  1. public final void acquire(int arg) {
  2. if (!tryAcquire(arg) &&
  3. acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
  4. selfInterrupt();
  5. }

代码虽然短,但包含的逻辑却很多,一步一步看下:

  1. 首先是调用开发人员自己实现的tryAcquire() 方法尝试获取锁资源,如果成功则整个acquire()方法执行完毕,即当前线程获得锁资源,可以进入临界区。
  2. 如果获取锁失败,则开始进入后面的逻辑,首先是addWaiter(Node.EXCLUSIVE)方法。来看下这个方法的源码实现:
  1. //注意:该入队方法的返回值就是新创建的节点
  2. private Node addWaiter(Node mode) {
  3. //基于当前线程,节点类型(Node.EXCLUSIVE)创建新的节点
  4. //由于这里是独占模式,因此节点类型就是Node.EXCLUSIVE
  5. Node node = new Node(Thread.currentThread(), mode);
  6. Node pred = tail;
  7. //这里为了提搞性能,首先执行一次快速入队操作,即直接尝试将新节点加入队尾
  8. if (pred != null) {
  9. node.prev = pred;
  10. //这里根据CAS的逻辑,即使并发操作也只能有一个线程成功并返回,其余的都要执行后面的入队操作。即enq()方法
  11. if (compareAndSetTail(pred, node)) {
  12. pred.next = node;
  13. return node;
  14. }
  15. }
  16. enq(node);
  17. return node;
  18. }
  19. //完整的入队操作
  20. private Node enq(final Node node) {
  21. for (;;) {
  22. Node t = tail;
  23. //如果队列还没有初始化,则进行初始化,即创建一个空的头节点
  24. if (t == null) {
  25. //同样是CAS,只有一个线程可以初始化头结点成功,其余的都要重复执行循环体
  26. if (compareAndSetHead(new Node()))
  27. tail = head;
  28. } else {
  29. //新创建的节点指向队列尾节点,毫无疑问并发情况下这里会有多个新创建的节点指向队列尾节点
  30. node.prev = t;
  31. //基于这一步的CAS,不管前一步有多少新节点都指向了尾节点,这一步只有一个能真正入队成功,其他的都必须重新执行循环体
  32. if (compareAndSetTail(t, node)) {
  33. t.next = node;
  34. //该循环体唯一退出的操作,就是入队成功(否则就要无限重试)
  35. return t;
  36. }
  37. }
  38. }
  39. }

上面的入队操作有两点需要说明:

一、初始化队列的触发条件就是当前已经有线程占有了锁资源,因此上面创建的空的头节点可以认为就是当前占有锁资源的节点(虽然它并没有设置任何属性)

二、注意整个代码是处在一个死循环中,知道入队成功。如果失败了就会不断进行重试。

  1. 经过上面的操作,我们申请获取锁的线程已经成功加入了等待队列,通过文章最一开始说的独占锁获取流程,那么节点现在要做的就是挂起当前线程,等待被唤醒,这个逻辑是怎么实现的呢?来看下源码:
  1. 通过上面的分析,该方法入参node就是刚入队的包含当前线程信息的节点
  2. final boolean acquireQueued(final Node node, int arg) {
  3. //锁资源获取失败标记位
  4. boolean failed = true;
  5. try {
  6. //等待线程被中断标记位
  7. boolean interrupted = false;
  8. //这个循环体执行的时机包括新节点入队和队列中等待节点被唤醒两个地方
  9. for (;;) {
  10. //获取当前节点的前置节点
  11. final Node p = node.predecessor();
  12. //如果前置节点就是头结点,则尝试获取锁资源
  13. if (p == head && tryAcquire(arg)) {
  14. //当前节点获得锁资源以后设置为头节点,这里继续理解我上面说的那句话
  15. //头结点就表示当前正占有锁资源的节点
  16. setHead(node);
  17. p.next = null; //帮助GC
  18. //表示锁资源成功获取,因此把failed置为false
  19. failed = false;
  20. //返回中断标记,表示当前节点是被正常唤醒还是被中断唤醒
  21. return interrupted;
  22. }
  23. 如果没有获取锁成功,则进入挂起逻辑
  24. if (shouldParkAfterFailedAcquire(p, node) &&
  25. parkAndCheckInterrupt())
  26. interrupted = true;
  27. }
  28. } finally {
  29. //最后会分析获取锁失败处理逻辑
  30. if (failed)
  31. cancelAcquire(node);
  32. }
  33. }

挂起逻辑是很重要的逻辑,这里拿出来单独分析一下,首先要注意目前为止,我们只是根据当前线程,节点类型创建了一个节点并加入队列中,其他属性都是默认值

  1. //首先说明一下参数,node是当前线程的节点,pred是它的前置节点
  2. private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {
  3. //获取前置节点的waitStatus
  4. int ws = pred.waitStatus;
  5. if (ws == Node.SIGNAL)
  6. //如果前置节点的waitStatus是Node.SIGNAL则返回true,然后会执行parkAndCheckInterrupt()方法进行挂起
  7. return true;
  8. if (ws > 0) {
  9. //由waitStatus的几个取值可以判断这里表示前置节点被取消
  10. do {
  11. node.prev = pred = pred.prev;
  12. } while (pred.waitStatus > 0);
  13. //这里我们由当前节点的前置节点开始,一直向前找最近的一个没有被取消的节点
  14. //注,由于头结点head是通过new Node()创建,它的waitStatus为0,因此这里不会出现空指针问题,也就是说最多就是找到头节点上面的循环就退出了
  15. pred.next = node;
  16. } else {
  17. //根据waitStatus的取值限定,这里waitStatus的值只能是0或者PROPAGATE,那么我们把前置节点的waitStatus设为Node.SIGNAL然后重新进入该方法进行判断
  18. compareAndSetWaitStatus(pred, ws, Node.SIGNAL);
  19. }
  20. return false;
  21. }

上面这个方法逻辑比较复杂,它是用来判断当前节点是否可以被挂起,也就是唤醒条件是否已经具备,即如果挂起了,那一定是可以由其他线程来唤醒的。该方法如果返回false,即挂起条件没有完备,那就会重新执行acquireQueued方法的循环体,进行重新判断,如果返回true,那就表示万事俱备,可以挂起了,就会进入parkAndCheckInterrupt()方法看下源码:

  1. private final boolean parkAndCheckInterrupt() {
  2. LockSupport.park(this);
  3. //被唤醒之后,返回中断标记,即如果是正常唤醒则返回false,如果是由于中断醒来,就返回true
  4. return Thread.interrupted();
  5. }

看acquireQueued方法中的源码,如果是因为中断醒来,那么就把中断标记置为true。不管是正常被唤醒还是由与中断醒来,都会去尝试获取锁资源。如果成功则返回中断标记,否则继续挂起等待。

注:Thread.interrupted()方法在返回中断标记的同时会清除中断标记,也就是说当由于中断醒来然后获取锁成功,那么整个acquireQueued方法就会返回true表示是因为中断醒来,但如果中断醒来以后没有获取到锁,继续挂起,由于这次的中断已经被清除了,下次如果是被正常唤醒,那么acquireQueued方法就会返回false,表示没有中断。

最后我们回到acquireQueued方法的最后一步,finally模块。这里是针对锁资源获取失败以后做的一些善后工作,翻看上面的代码,其实能进入这里的就是tryAcquire()方法抛出异常,也就是说AQS框架针对开发人员自己实现的获取锁操作如果抛出异常,也做了妥善的处理,一起来看下源码:

  1. //传入的方法参数是当前获取锁资源失败的节点
  2. private void cancelAcquire(Node node) {
  3. // 如果节点不存在则直接忽略
  4. if (node == null)
  5. return;
  6. node.thread = null;
  7. // 跳过所有已经取消的前置节点,跟上面的那段跳转逻辑类似
  8. Node pred = node.prev;
  9. while (pred.waitStatus > 0)
  10. node.prev = pred = pred.prev;
  11. //这个是前置节点的后继节点,由于上面可能的跳节点的操作,所以这里可不一定就是当前节点,仔细想一下。^_^
  12. Node predNext = pred.next;
  13. //把当前节点waitStatus置为取消,这样别的节点在处理时就会跳过该节点
  14. node.waitStatus = Node.CANCELLED;
  15. //如果当前是尾节点,则直接删除,即出队
  16. //注:这里不用关心CAS失败,因为即使并发导致失败,该节点也已经被成功删除
  17. if (node == tail && compareAndSetTail(node, pred)) {
  18. compareAndSetNext(pred, predNext, null);
  19. } else {
  20. int ws;
  21. if (pred != head &&
  22. ((ws = pred.waitStatus) == Node.SIGNAL ||
  23. (ws <= 0 && compareAndSetWaitStatus(pred, ws, Node.SIGNAL))) &&
  24. pred.thread != null) {
  25. Node next = node.next;
  26. if (next != null && next.waitStatus <= 0)
  27. //这里的判断逻辑很绕,具体就是如果当前节点的前置节点不是头节点且它后面的节点等待它唤醒(waitStatus小于0),
  28. //再加上如果当前节点的后继节点没有被取消就把前置节点跟后置节点进行连接,相当于删除了当前节点
  29. compareAndSetNext(pred, predNext, next);
  30. } else {
  31. //进入这里,要么当前节点的前置节点是头结点,要么前置节点的waitStatus是PROPAGATE,直接唤醒当前节点的后继节点
  32. unparkSuccessor(node);
  33. }
  34. node.next = node; // help GC
  35. }
  36. }

上面就是独占模式获取锁的核心源码,确实非常难懂,很绕,就这几个方法需要反反复复看很多遍,才能慢慢理解。

接下来看下释放锁的过程:

  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. }

tryRelease()方法是用户自定义的释放锁逻辑,如果成功,就判断等待队列中有没有需要被唤醒的节点(waitStatus为0表示没有需要被唤醒的节点),一起看下唤醒操作:

  1. private void unparkSuccessor(Node node) {
  2. int ws = node.waitStatus;
  3. if (ws < 0)
  4. //把标记为设置为0,表示唤醒操作已经开始进行,提高并发环境下性能
  5. compareAndSetWaitStatus(node, ws, 0);
  6. Node s = node.next;
  7. //如果当前节点的后继节点为null,或者已经被取消
  8. if (s == null || s.waitStatus > 0) {
  9. s = null;
  10. //注意这个循环没有break,也就是说它是从后往前找,一直找到离当前节点最近的一个等待唤醒的节点
  11. for (Node t = tail; t != null && t != node; t = t.prev)
  12. if (t.waitStatus <= 0)
  13. s = t;
  14. }
  15. //执行唤醒操作
  16. if (s != null)
  17. LockSupport.unpark(s.thread);
  18. }

相比而言,锁的释放操作就简单很多了,代码也比较少。

三、总结

以上就是AQS独占锁的获取与释放过程,大致思想很简单,就是尝试去获取锁,如果失败就加入一个队列中挂起。释放锁时,如果队列中有等待的线程就进行唤醒。但如果一步一步看源码,会发现细节非常多,很多地方很难搞明白,我自己也是反反复复学习很久才有点心得,但也不敢说已经研究通了AQS,甚至不敢说我上面的研究成果就是对的,只是写篇文章总结一下,跟同行交流交流心得。

除了独占锁,后面还会产出AQS一系列的文章,包括共享锁,条件队列的实现原理等。

深入浅出AQS之独占锁模式的更多相关文章

  1. AQS详解之独占锁模式

    AQS介绍 AbstractQueuedSynchronizer简称AQS,即队列同步器.它是JUC包下面的核心组件,它的主要使用方式是继承,子类通过继承AQS,并实现它的抽象方法来管理同步状态,它分 ...

  2. 深入浅出AQS之共享锁模式

    在了解了AQS独占锁模式以后,接下来再来看看共享锁的实现原理. 原文地址:http://www.jianshu.com/p/1161d33fc1d0 搞清楚AQS独占锁的实现原理之后,再看共享锁的实现 ...

  3. 深入浅出AQS之条件队列

    相比于独占锁跟共享锁,AbstractQueuedSynchronizer中的条件队列可能被关注的并不是很多,但它在阻塞队列的实现里起着至关重要的作用,同时如果想全面了解AQS,条件队列也是必须要学习 ...

  4. 深入浅出AQS之组件概览

    之前分析了AQS中的独占锁,共享锁,条件队列三大模块,现在从结构上来看看AQS各个组件的情况. 原文地址:http://www.jianshu.com/p/49b86f9cd7ab 深入浅出AQS之独 ...

  5. 关于AQS——独占锁特性+共享锁实现(二)

    五.可中断获取锁的实现(独占锁的特性之一) 我们知道lock相较于synchronized有一些更方便的特性,比如能响应中断以及超时等待等特性,现在我们依旧采用通过学习源码的方式来看看能够响应中断是怎 ...

  6. 深入理解Java并发框架AQS系列(三):独占锁(Exclusive Lock)

    一.前言 优秀的源码就在那里 经过了前面两章的铺垫,终于要切入正题了,本章也是整个AQS的核心之一 从本章开始,我们要精读AQS源码,在欣赏它的同时也要学会质疑它.当然本文不会带着大家逐行过源码(会有 ...

  7. 深入浅出Java并发包—锁机制(二)

    接上文<深入浅出Java并发包—锁机制(一)  >  2.Sync.FairSync.TryAcquire(公平锁) 我们直接来看代码 protected final boolean tr ...

  8. JUC之AbstractQueuedSynchronizer原理分析 - 独占/共享模式

    1. 简介 AbstractQueuedSynchronizer (抽象队列同步器,以下简称 AQS)出现在 JDK 1.5 中,由大师 Doug Lea 所创作.AQS 是很多同步器的基础框架. R ...

  9. 深入浅出 Java Concurrency (8): 锁机制 part 3[转]

    接上篇,这篇从Lock.lock/unlock开始.特别说明在没有特殊情况下所有程序.API.文档都是基于JDK 6.0的. public void java.util.concurrent.lock ...

随机推荐

  1. 《Java从入门到放弃》JavaSE篇:综合练习——单身狗租赁系统(数组版)

    因为现在只学习了基本语法,所以在综合练习之前,先补充关于方法概念. 方法的作用:把一系列的代码放在一起,然后再取个别名.之后通过这个别名的调用,就相当于执行了这一系列的代码. 方法的语法:([]中的内 ...

  2. python+selenium自动化软件测试(第8章) :多线程

    前戏:线程的基础 运行多个线程同时运行几个不同的程序类似,但具有以下优点:进程内共享多线程与主线程相同的数据空间,如果他们是独立的进程,可以共享信息或互相沟通更容易.线程有时称为轻量级进程,他们并不需 ...

  3. 大手册(书籍)排版利器-XML自动排版生成工具

    --支持全球化/多语言/符合W3C标准的XML自动排版工具 Boxth XML/XSL Formatter是专为XML数据或其他结构化数据源自动输出排版文件(如: PDF等)而设计的集数据格式化.版式 ...

  4. Python之set

    set set集合,是一个无序且不重复的元素集合 set的优势 set 的访问数度快 set 原生解决数据重复问题 # 数据库中原有 old_dict = { "#1":{ 'ho ...

  5. HTML的基本标签

    整理一下这一周学习的一些知识. 首先是一些基本标签. <!DOCTYPE HTML><html> 文档类型声明: 让浏览器,按照html5的标准对代码进行解释与执行.文档类型声 ...

  6. Microsoft Dynamics 365 之 味全食品 项目分享和Customer Engagement新特性分享

    味全食品 Dynamics 365项目: 在企业门户和电子商务等新营销模式频出的今天,零售业需要利用统一的管理平台管理日益庞大的客户及销售数据,整合线上线下的零售业务,从采购.仓储.生产.配送到销售. ...

  7. fixed定位兼容性

    不过从ios5.1以来,fixed定位就已经支持了,但很遗憾,ios现在对它还只是半支持. 但是在某些情况下,会出现一些比较奇葩的问题,比如fixed元素中存在输入框子元素,这个时候就会跪了. 可以看 ...

  8. 在tomcat7中启用HTTPS的详细配置

    详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt385 最简单的方法,直接用java里的keytool工具生成一个keysto ...

  9. 汇编指令-CMP、TEQ(5)

     cmp:(compare)指令进行比较两个操作数的大小  格式: cmp oprd1,oprd2 比较oprd1和oprd2操作数,然后通过助记符来实现想要的判断. teq: (test equal ...

  10. Socket通信中AF_INET 和 AF_UNIX域的区别

    转载:http://blog.csdn.net/sandware/article/details/40923491 1.  AF_INET域socket通信过程 典型的TCP/IP四层模型的通信过程. ...