之前分析了AQS中的独占锁,共享锁,条件队列三大模块,现在从结构上来看看AQS各个组件的情况。

原文地址:http://www.jianshu.com/p/49b86f9cd7ab

深入浅出AQS之独占锁模式

深入浅出AQS之共享锁模式

深入浅出AQS之条件队列

前面三篇文章如果之前没有AQS基础的话看起来会比较吃力,这篇文章说明一下AQS的基础知识,方便快速了解AQS。

首先AQS的基本执行过程就是尝试获取锁,成功则返回,如果失败就进入同步队列进行锁资源的等待。基于这个流程可以看出队列跟队列中的节点应该是两个重点。

首先来看下AQS里队列节点Node的结构:



该类中有五个字段,依次来看一下:

  1. prev,next:指向它的前置节点跟后继节点,由此看出AQS中的同步队列是个双向链表。
  2. thread:当前线程对象。
  3. waitStatus:当前节点的状态,是个int类型变量,依次有如下几种:
类型 说明
-1 SIGNAL 当前节点的后继节点被阻塞,因此当当前节点在释放或者取消的时候需要唤醒它的后继节点。
1 CANCELLED 当前节点由于超时或者中断被取消,节点进入这个状态以后将保持不变。注:这是唯一大于0的值,很多判断逻辑会用到这个特征
-2 CONDITION 当前节点正处在条件队列中,在条件达成前不能获取锁。
-3 PROPAGATE 当前节点获取到锁的信息需要传递给后继节点,共享锁模式使用该值。
0 节点初始状态。
  1. nextWaiter:如果当前节点是共享模式,该值会指向一个SHARE节点。如果当前节点是在条件队列中,则该值会指向下一个等待条件的节点。

了解了Node节点的数据结构以后,看下独占锁模式下的同步队列的结构:



注:head节点是new出来一个新的Node节点,而tail是直接指向队列中最后一个节点。

了解了独占锁模式队列以后,看下共享锁模式下的同步队列(注意对比其中的不同):



注:共享锁跟独占锁是同一个同步队列,也就是说同步队列中的节点既可以是共享类型也可以是独占类型。

除了独占锁跟共享锁使用的同步队列,还有一个很重要的队列就是条件队列,一起看下:

注意区分条件队列跟同步队列的区别:1、头尾指针,2、单链表

搞明白了AQS中这些基础的数据结构以后,最后再看下AQS对外提供的API:

独占锁模式:

方法名 说明
acquire 独占模式获取锁,不响应中断,如果发生中断只会把当前线程的中断状态设置为true
acquireInterruptibly 独占模式获取锁,如果在获取锁的过程中线程被中断,则直接抛出中断异常
release 释放锁资源

共享锁模式:

方法名 说明
acquireShared 共享模式获取锁,不响应中断,如果发生中断只会把当前线程的中断状态设置为true
acquireSharedInterruptibly 共享模式获取锁,如果在获取锁的过程中线程被中断,则直接抛出中断异常
releaseShared 释放锁资源

条件队列:

方法名 说明
await 阻塞等待条件,如果被中断则抛出中断异常
awaitUninterruptibly 阻塞等待条件,不响应中断,如果发生中断只会把线程中断状态设置为true
awaitNanos 等待纳秒时间,如果被中断则抛出中断异常
awaitUntil 等待直到一个截止时间,如果被中断则抛出中断异常
await(long time, TimeUnit unit) 等待一个指定时间,如果被中断则抛出中断异常
signal 唤醒等待队列中的第一个节点
signalAll 唤醒等待队列中的所有节点

扩展API:

方法名 说明
tryAcquire 尝试获取独占锁,不阻塞
tryAcquireNanos 尝试在指定纳秒时间内获取独占锁,如果被中断则抛出中断异常
tryRelease 尝试释放独占锁,不阻塞
tryAcquireShared 尝试获取共享锁,不阻塞
tryAcquireSharedNanos 尝试在指定纳秒时间内获取共享锁,如果被中断则抛出中断异常
tryReleaseShared 尝试释放共享锁,不阻塞

了解了上面介绍的关于AQS的基础数据结构及API以后,再去看我之前写的三篇分别介绍独占锁,共享锁,条件队列实现原理的文章就不会云里雾里了。

深入浅出AQS之组件概览的更多相关文章

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

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

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

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

  3. AQS同步组件及ReentrantLock和synchronized的区别

    AQS同步组件 CountDownLatch(只有一个线程对他进行操作): 主线程必须在启动其它线程后立即调用await()方法.这样主线程的操作就会在这个方法上阻塞,直到其它线程完成各自的任务. S ...

  4. AQS 原理以及 AQS 同步组件总结

    1 AQS 简单介绍 AQS 的全称为(AbstractQueuedSynchronizer),这个类在 java.util.concurrent.locks 包下面. AQS 是一个用来构建锁和同步 ...

  5. 并发编程-深入浅出AQS

    AQS是并发编程中非常重要的概念,它是juc包下的许多并发工具类,如CountdownLatch,CyclicBarrier,Semaphore 和锁, 如ReentrantLock, ReaderW ...

  6. 一文搞懂AQS及其组件的核心原理

    @ 目录 前言 AbstractQueuedSynchronizer Lock ReentrantLock 加锁 非公平锁/公平锁 lock tryAcquire addWaiter acquireQ ...

  7. 从零开始了解多线程 之 深入浅出AQS -- 上

    java锁&AQS深入浅出学习--上 上一篇文章中我们一起学习了jvm缓存一致性.多线程间的原子性.有序性.指令重排的相关内容, 这一篇文章便开始和大家一起学习学习AQS(AbstractQu ...

  8. 深入浅出AQS之独占锁模式

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

  9. 【java】Java组件概览(2)— 基本库

    1.Math Math相关的库包括包括浮点库(java.lang.Math和java.lang.StrictMath)和任意精度数学(java.math包). (1)java.lang.Math 该类 ...

随机推荐

  1. VHDL学习:利用Quartus自带库3步快速完成状态机

    Quartus自带库里面有各种编程语言的模板,供开发者参考. 初学者利用VHDL实现状态机比较生疏的情况下,可以调出该模板,适当修改即可. 本文将描述如何利用Quartus自带库调出状态机模板,并适当 ...

  2. 前端AES的加密和解密

    在工作的过程中,经常要对一些数据做一些加密,当然有复杂的加密和简单的加密,也有对称加密等等.总之加密的方式有很多.今天在这里,我只是简单的分享一个我最近遇到的加密方式-AES.这个也是后端工程师用的, ...

  3. MongoDB学习之路(一)

    NoSQL简介 NoSQL(Not Only SQL),意为"不仅仅是SQL" 关系型数据库遵循ACID规则 1. A(Atomicity)原子性 指的是事务里的所有操作要么全部做 ...

  4. [2017BUAA软工助教]博客格式的详细说明

    一.为什么要强调博客格式 可以对比粗读一下这几篇博客然后自己感受一下博客格式对博客阅读体验的影响: MarkDown流:    [schaepher]2017春季 JMU 1414软工助教 链接汇总 ...

  5. 201521123072《java程序设计》第四次总结

    1. 本周学习总结 1.1 尝试使用思维导图总结有关继承的知识点. 1.11.2 使用常规方法总结其他上课内容 一些小的方法归纳: 通过 instanceof 可以判断父类引用所引用的对象实例的实际类 ...

  6. PTA中提交Java程序的一些套路

    201708新版改版说明 PTA与2017年8月已升级成新版,域名改为https://pintia.cn/,官方建议使用Firefox与Chrome浏览器. 旧版 PTA 用户首次在新版系统登录时,请 ...

  7. 201521123106 《Java程序设计》第13周学习总结

    1. 本周学习总结 以你喜欢的方式(思维导图.OneNote或其他)归纳总结多网络相关内容. 2. 书面作业 1. 网络基础 1.1 比较ping www.baidu.com与ping cec.jmu ...

  8. 201521123110《Java程序设计》第14周学习总结

    1. 本周学习总结 2. 书面作业 1. MySQL数据库基本操作 2. 使用JDBC连接数据库与Statement 2.1 使用Statement操作数据库.(粘贴一段你认为比较有价值的代码,出现学 ...

  9. 201521123068《Java程序设计》第12周学习总结

    1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多流与文件相关内容. 2. 书面作业 将Student对象(属性:int id, String name,int age,doubl ...

  10. 201521123081《Java程序设计》 第10周学习总结

    1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结异常与多线程相关内容. 参考资料:XMind 2. 书面作业 本次PTA作业题集异常.多线程. Q1. finally 题目4-2 ...