范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文
国学影视

面试官竟然问我Java的AQS锁实现原理,幸亏我看了底层源码

  我们常见的并发锁 ReentrantLock 、 CountDownLatch 、 Semaphore 、 CyclicBarrier 都是基于 AQS 实现的,所以说不懂 AQS 实现原理的,就不能说了解Java锁。
  上篇文章讲了AQS的加锁流程,这篇文章再一块看一下AQS具体源码实现。
  先回顾一下AQS的加锁流程 1. AQS加锁流程
  AQS 的加锁流程并不复杂,只要理解了 同步队列 和 条件队列 ,以及它们之间的数据流转,就算彻底理解了 AQS 。 当多个线程竞争AQS锁时,如果有个线程获取到锁,就把ower线程设置为自己 没有竞争到锁的线程,在同步队列中阻塞(同步队列采用双向链表,尾插法)。 持有锁的线程调用await方法,释放锁,追加到条件队列的末尾(条件队列采用单链表,尾插法)。 持有锁的线程调用signal方法,唤醒条件队列的头节点,并转移到同步队列的末尾。 同步队列的头节点优先获取到锁
  了解AQS加锁流程之后,再去看源码就容易理解了。 2. AQS的数据结构// 继承自AbstractOwnableSynchronizer,为了记录哪个线程占用锁 public abstract class AbstractQueuedSynchronizer extends AbstractOwnableSynchronizer {        // 同步状态,0表示无锁,每次加锁+1,释放锁-1     private volatile int state;      // 同步队列的头尾节点     private transient volatile Node head;     private transient volatile Node tail;      // Node节点,用来包装线程,放到队列中     static final class Node {         // 节点中的线程         volatile Thread thread;          // 节点状态         volatile int waitStatus;          // 同步队列的前驱节点和后继节点         volatile Node prev;         volatile Node next;          // 条件队列的后继节点         Node nextWaiter;     }      // 条件队列     public class ConditionObject implements Condition {         // 条件队列的头尾节点         private transient Node firstWaiter;         private transient Node lastWaiter;     } }
  首先AQS继承自AbstractOwnableSynchronizer,其实是为了记录哪个线程正在占用锁。 public abstract class AbstractOwnableSynchronizer {      private transient Thread exclusiveOwnerThread;      // 设置占用锁的线程     protected final void setExclusiveOwnerThread(Thread thread) {         exclusiveOwnerThread = thread;     }      protected final Thread getExclusiveOwnerThread() {         return exclusiveOwnerThread;     } }
  无论是同步队列还是条件队列中线程都需要包装成Node节点。
  虽然同步队列和条件队列都是由Node节点组成的,但是同步队列中是使用prev和next组成双向链表,nextWaiter只用来表示是共享模式还是排他模式。
  条件队列没有使用到Node中prev和next属性,而是使用nextWaiter组成单链表。
  这个复用对象的设计思想值得我们学习。
  同步队列head节点是个哑节点,里面并没有存储线程对象。当然head节点也可以看成是给当前持有锁的线程使用的。
  Node节点的状态(waitStatus)共有5种: 1 cancelled:表示线程已经被取消 0 初始化:Node节点的默认值 -1 signal: 表示节点线程在释放锁后要唤醒同步队列中的下一个节点线程 -2 condition: 当前节点在条件队列中 -3 propagate: 释放共享资源的时候会向后传播释放其他共享节点(用于共享模式) 3. AQS方法概览
  AQS支持独占和共享两种访问资源的模式(独占模式又叫排他模式)。
  独占模式的方法: // 加锁 acquire(); // 加可中断的锁 acquireInterruptibly(); // 一段时间内,加锁不成功,就不加了 tryAcquireNanos(int arg, long nanosTimeout); // 释放锁 release();
  共享模式的方法: // 加锁 acquireShared(); // 加可中断的锁 acquireSharedInterruptibly(); // 一段时间内,加锁不成功,就不加了 tryAcquireSharedNanos(int arg, long nanosTimeout); // 释放锁 releaseShared();
  独占模式和共享模式的方法并没有实现具体的加锁、释放锁逻辑,AQS中只是定义了加锁、释放锁的抽象方法。
  留给子类实现的抽象方法: // 加独占锁 protected boolean tryAcquire(int arg) {     throw new UnsupportedOperationException(); } // 释放独占锁 protected boolean tryRelease(int arg) {     throw new UnsupportedOperationException(); }  // 加共享锁 protected int tryAcquireShared(int arg) {     throw new UnsupportedOperationException(); } // 释放共享锁 protected boolean tryReleaseShared(int arg) {     throw new UnsupportedOperationException(); }  // 判断是否是当前线程正在持有锁 protected boolean isHeldExclusively() {     throw new UnsupportedOperationException(); }
  这里就用到了设计模式中的模板模式,父类AQS定义了加锁、释放锁的流程,子类 ReentrantLock 、 CountDownLatch 、 Semaphore 、 CyclicBarrier 负责实现具体的加锁、释放锁逻辑。
  这不是个面试知识点吗?
  面试官再问你,你看过哪些框架源码使用到了设计模式?
  你就可以回答AQS源码中用到了模板模式,巴拉巴拉,妥妥的加分项!
  4. AQS源码剖析
  整个加锁流程如下:
  先看一下加锁方法的源码: 4.1 加锁// 加锁方法,传参是1 public final void acquire(int arg) {     // 1. 首先尝试获取锁,如果获取成功,则设置state+1,exclusiveOwnerThread=currentThread(留给子类实现)     if (!tryAcquire(arg) &&             // 2. 如果没有获取成功,把线程组装成Node节点,追加到同步队列末尾             acquireQueued(addWaiter(Node.EXCLUSIVE), arg)) {         // 3. 加入同步队列后,将自己挂起         selfInterrupt();     } }
  再看一下addWaiter方法源码,作用就是把线程组装成Node节点,追加到同步队列末尾。 // 追加到同步队列末尾,传参是共享模式or排他模式 private Node addWaiter(Node mode) {     // 1. 组装成Node节点     Node node = new Node(Thread.currentThread(), mode);     Node pred = tail;     if (pred != null) {         node.prev = pred;         // 2. 在多线程竞争不激烈的情况下,通过CAS方法追加到同步队列末尾         if (compareAndSetTail(pred, node)) {             pred.next = node;             return node;         }     }     // 3. 在多线程竞争激烈的情况下,使用死循环保证追加到同步队列末尾     enq(node);     return node; }  // 创建Node节点,传参是线程,共享模式or排他模式 Node(Thread thread, Node mode) {     this.thread = thread;     this.nextWaiter = mode; }  // 通过死循环的方式,追加到同步队列末尾 private Node enq(final Node node) {     for (; ; ) {         Node t = tail;         if (t == null) {             if (compareAndSetHead(new Node()))                 tail = head;         } else {             node.prev = t;             if (compareAndSetTail(t, node)) {                 t.next = node;                 return t;             }         }     } }
  再看一下addWaiter方法外层的acquireQueued方法,作用就是: 在追加到同步队列末尾后,再判断一下前驱节点是不是头节点。如果是,说明是第一个加入同步队列的,就再去尝试获取锁。 如果获取锁成功,就把自己设置成头节点。 如果前驱节点不是头节点,或者获取锁失败,就逆序遍历同步队列,找到可以将自己唤醒的节点。 最后才放心地将自己挂起 // 追加到同步队列末尾后,再次尝试获取锁 final boolean acquireQueued(final Node node, int arg) {     boolean failed = true;     try {         boolean interrupted = false;         for (; ; ) {             // 1. 找到前驱节点             final Node p = node.predecessor();             // 2. 如果前驱节点是头结点,就再次尝试获取锁             if (p == head && tryAcquire(arg)) {                 // 3. 获取锁成功后,把自己设置为头节点                 setHead(node);                 p.next = null;                 failed = false;                 return interrupted;             }             // 4. 如果还是没有获取到锁,找到可以将自己唤醒的节点             if (shouldParkAfterFailedAcquire(p, node) &&                     // 5. 最后才放心地将自己挂起                     parkAndCheckInterrupt())                 interrupted = true;         }     } finally {         if (failed)             cancelAcquire(node);     } }
  再看一下shouldParkAfterFailedAcquire方法,是怎么找到将自己唤醒的节点的?为什么要找这个节点? // 加入同步队列后,找到能将自己唤醒的节点 private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {     int ws = pred.waitStatus;     // 1. 如果前驱节点的状态已经是SIGNAL状态(释放锁后,需要唤醒后继节点),就无需操作了     if (ws == Node.SIGNAL)         return true;     // 2. 如果前驱节点的状态是已取消,就继续向前遍历     if (ws > 0) {         do {             node.prev = pred = pred.prev;         } while (pred.waitStatus > 0);         pred.next = node;     } else {         // 3. 找到了不是取消状态的节点,把该节点状态设置成SIGNAL         compareAndSetWaitStatus(pred, ws, Node.SIGNAL);     }     return false; }
  从代码中可以很清楚的看到,目的就是为了找到不是取消状态的节点,并把该节点的状态设置成SIGNAL。
  状态是SIGNAL的节点,释放锁后,需要唤醒其后继节点。
  简单理解就是:小弟初来乍到,特意来知会老大一声,有好事,多通知小弟。
  再看一下释放锁的逻辑。 4.2 释放锁
  释放锁的流程如下:
  释放锁的代码逻辑比较简单: // 释放锁 public final boolean release(int arg) {     // 1. 先尝试释放锁,如果时候成功,则设置state-1,exclusiveOwnerThread=null(由子类实现)     if (tryRelease(arg)) {         Node h = head;         // 2. 如果同步队列中还有其他节点,就唤醒下一个节点         if (h != null && h.waitStatus != 0)             // 3. 唤醒其后继节点             unparkSuccessor(h);         return true;     }     return false; }
  再看一下唤醒后继节点的方法 // 唤醒后继节点 private void unparkSuccessor(Node node) {     int ws = node.waitStatus;     // 1. 如果头节点不是取消状态,就重置成初始状态     if (ws < 0)         compareAndSetWaitStatus(node, ws, 0);      Node s = node.next;     // 2. 如果后继节点是null或者是取消状态     if (s == null || s.waitStatus > 0) {         s = null;         // 3. 从队尾开始遍历,找到一个有效状态的节点         for (Node t = tail; t != null && t != node; t = t.prev)             if (t.waitStatus <= 0)                 s = t;     }     // 3. 唤醒这个有效节点     if (s != null)         LockSupport.unpark(s.thread); }4.3 await等待
  await等待的流程:
  持有锁的线程可以调用await方法,作用是:释放锁,并追加到条件队列末尾。 // 等待方法 public final void await() throws InterruptedException {     // 如果线程已中断,则中断     if (Thread.interrupted())         throw new InterruptedException();     // 1. 追加到条件队列末尾     Node node = addConditionWaiter();     // 2. 释放锁     int savedState = fullyRelease(node);     int interruptMode = 0;     // 3. 有可能刚加入条件队列就被转移到同步队列了,如果还在条件队列,就可以放心地挂起自己     while (!isOnSyncQueue(node)) {         LockSupport.park(this);         if ((interruptMode = checkInterruptWhileWaiting(node)) != 0)             break;     }     // 4. 如果已经转移到同步队列,就尝试获取锁     if (acquireQueued(node, savedState) && interruptMode != THROW_IE)         interruptMode = REINTERRUPT;     if (node.nextWaiter != null)         // 5. 清除条件队列中已取消的节点         unlinkCancelledWaiters();     if (interruptMode != 0)         reportInterruptAfterWait(interruptMode); }
  再看一下addConditionWaiter方法,是怎么追加到条件队列末尾的? // 追加到条件队列末尾 private Node addConditionWaiter() {     Node t = lastWaiter;     // 1. 清除已取消的节点,找到有效节点     if (t != null && t.waitStatus != Node.CONDITION) {         unlinkCancelledWaiters();         t = lastWaiter;     }     // 2. 创建Node节点,状态是-2(表示处于条件队列)     Node node = new Node(Thread.currentThread(), Node.CONDITION);     // 3. 追加到条件队列末尾     if (t == null)         firstWaiter = node;     else         t.nextWaiter = node;     lastWaiter = node;     return node; }4.4 signal唤醒
  signal唤醒的流程:
  唤醒条件队列的头节点,并追加到同步队列末尾。 // 唤醒条件队列的头节点 public final void signal() {     // 1. 只有持有锁的线程才能调用signal方法     if (!isHeldExclusively())         throw new IllegalMonitorStateException();     // 2. 找到条件队列的头节点     Node first = firstWaiter;     if (first != null)         // 3. 开始唤醒         doSignal(first); }  // 实际的唤醒方法 private void doSignal(Node first) {     do {         // 4. 从条件队列中移除头节点         if ((firstWaiter = first.nextWaiter) == null)             lastWaiter = null;         first.nextWaiter = null;         // 5. 使用死循环,一定要转移一个节点到同步队列     } while (!transferForSignal(first) &&             (first = firstWaiter) != null); }
  到底是怎么转移到同步队列末尾的? // 实际转移方法 final boolean transferForSignal(Node node) {     // 1. 把节点状态从CONDITION改成0     if (!compareAndSetWaitStatus(node, Node.CONDITION, 0))         return false;      // 2. 使用死循环的方式,追加到同步队列末尾(前面已经讲过)     Node p = enq(node);     int ws = p.waitStatus;     // 3. 把前驱节点状态设置SIGNAL(通知他,别忘了唤醒老弟)     if (ws > 0 || !compareAndSetWaitStatus(p, ws, Node.SIGNAL))         LockSupport.unpark(node.thread);     return true; }5. 总结
  看完整个AQS的源码,是不是完全理解了AQS加锁、释放锁、以及同步队列和条件队列数据流转的逻辑了。
  连AQS这么复杂的源码你都搞清楚了,下篇带你一块学习ReentrantLock源码,应该就轻松多了。
  我是「一灯架构」,如果本文对你有帮助,欢迎各位小伙伴点赞、评论和关注,感谢各位老铁,我们下期见

达摩沙漠僧皮肤官宣,兰陵王皮肤被抢,排位赛回归奖励公布爱生活,爱游戏,大家好,我是阿呆。期待大家的关注,我会在这里分享更多有趣的最新资讯。前言对于王者荣耀的S26赛季,相信大家近期应该都有了解过,一般情况下,新赛季上线将会迎来大量新皮TBC怀旧服海山无限发财团,想赚钱的不容错过最近几天更新了一波海山的攻略,收到粉丝的不少反馈,好几个粉丝都对海山小怪团很感兴趣,今天详细聊一聊。首先,我可以负责任地说,海山小怪一定可以无限刷,因为这是副本的机制决定的,除非暴韶山偶遇一个老兵的赤诚今年十月,在参观韶山毛泽东故居的旅途上认识了他。高高的个子,头发花白,红润的脸膛,浓浓的眉毛,挺直的腰背,背旅游包,着旅游鞋,颇有精神。韶山南站下高铁,我不受包车师傅的忽悠,坚持乘跨年行情,白酒券商能上涨到多高?1211周末闲聊从12月5号开始,一直到14号,头条推出了生机大会的活动。作为k线妖的粉丝,多年来一直习惯了开盘前看一眼每天的早盘碎片,对当天的大盘做到心中有数。现在,k线妖需要你去今日头条k线妖中国A股能与宁德媲美的这5家新能源科技企业,未来可期A股市场上,能够代表未来十年的新能源硬核科技,并且还能与宁德时代过去十年谷价涨幅媲美的高端制造骨干,这5将新能源科技企业绝对可以当选。虽然呢,全网上号称是高端制造的公司数不胜数,但氢能源概念股票有哪些?氢能源股票信息一览表氢能源是国家未来大力发展的方向之一,是十四五规划的重点。氢能是公认的清洁能源,作为低碳和零碳能源正在脱颖而出。21世纪,我国和美国日本加拿大欧盟等都制定了氢能发展规划,并且我国已在锂电池和新能源股票三年内都没有投资价值人声鼎沸之际,就是散场之时。你应该在最喧闹时离席,席下的那些波诡云谲终将浮现。终了,一切就都是残局,都是惋惜和悔恨。锂电池行业已经出现严重产能过剩。新能源汽车行业迎来了发展的黄金期大盘能到3700点吗?喝酒吃药的行情真的来了吗?(附板块分析)首发公众号丨定投者笔记文丨韭菜妹转载请注明出处大家周末好!今天我们把本周A股的表现作一个简单的回顾,然后把各个板块作一个详细分析。01本周回顾三大指数受降准利好刺激,A股整体表现较3万座油站入局换电,中石化要新能源车企交出半条命?原创首发金角财经(FJinjiao)作者小迪一台车省7万30秒换上满电电池补贴不受价格限制换电,新能源汽车补能的理想模式,在政策企业和巨头的助推下,正在加速从B端走向C端。最近,三拜登推新能源车税收抵免,何以在国际上犯众怒?美国2022年开始实施新能源车税收抵免。图ICphoto近日,美国众议院通过了拜登总统的重建更好未来法案(BuildBackBetterAct),该法案的一项重要内容,是2022年如果给我三天假期,那就让美好不期而遇金秋十月,如果给你几天悠闲时光,你会选择去哪里消遣呢?如果你没有好想法的话?不妨来看看今日介绍的这个好去处在浦东川沙新镇,坐落着这样一栋小别墅推开木质大门,满目悠然的假日气息。如果
资讯加速布局储能业务,国轩高科491兆瓦时海外储能项目量产发货文懂车帝原创魏微懂车帝原创行业近日,懂车帝从官方了解到,国轩高科491兆瓦时(MWh)海外大型储能电站首批集装箱正式发货。据悉,此次交付的大型储能集装箱采用国轩高科自主研发和生产的河南省12月最新一批项目工程清单出炉!来看哪个项目投资额最大?经河南省政府同意,印发河南省2022年重点项目实施方案,聚焦园林景区绿化石油化学工程环保等10大重点领域,项目清单如下1。年产10000吨石墨制品项目投资额5000万元建设单位河南腾讯天美又放大招,和微软联手打造重返帝国,上架首日排行榜第一在众多游戏类型中,SLG可以说是久经不衰的代表,游戏融合了社交策略休闲战争养成,等多元素,可谓相当耐玩,而且玩家稳定,收入更稳定,像网易阿里莉莉丝等,旗下都有SLG手游常驻畅销榜,新华全媒科学家找到调控水稻小麦穗发芽的开关农作物在种子成熟期,如遇连阴雨不能及时收获,常会出现部分籽粒在穗上发芽的现象。穗发芽严重影响作物的产量和品质。我国科学家日前找到了调控水稻小麦穗发芽问题的一对开关,有望为避免穗发芽痛别!著名力学科学家黄克智院士逝世人民网北京12月7日电记者从清华大学获悉,中国共产党党员中国科学院院士清华大学航天航空学院工程力学系教授黄克智,因病医治无效,于2022年12月6日在北京逝世,享年95岁。黄克智院海信电视机质量怎么样?实力与性价比的明智之选作为国内名列前茅的电视机品牌,海信电视机每件都在不断推出新的型号来适应不同用户的需求,力求提升自身的技术来跟上时代的脚步。只不过很多普通老百姓可能对于这款海信电视机的相关具体信息还心情烦躁无助的时候你会怎么做心情烦躁,无助的时候你会怎么做?无端发火,焦躁不安,无助失望,想发火又无处可发,是今天一天的心情标签于是到了晚上我刷到人民日报的每日金句,执笔摘抄到我的本子上了,写着写着就静下心来国产对讲机也被造假?粉丝寄来假冒欧讯对讲机,大家一起来找茬今天咱们来聊一聊对讲机真伪识别的话题,这里说的真伪识别并不是那种贴个假牌子,而是从外形到细节都非常接近真机的复刻版对讲机。记得刚接触对讲机那会,市面上比较出名的应该就是八重洲的FT刘卫平人生的感悟文刘卫平前几年网上曾经流传了一首诗,我读来感概万千的,此诗表达了一生走过来的感受,回头来看,经历了那么多风风雨雨,再看晚年,生命在岁月中继续延续,只是对人生明了许多。我反复读了几遍谁的人生不委屈你的委屈来自于你的忿忿不平心犹不甘,而时代的跌宕起伏从来都不会考虑半点个人好恶。抛开所有不平不甘,只有你经历过炼狱般的自我磨砺,才能让心灵拥有创造世外桃源的力量。题记01。hr中午电视怎么选?认准三买三不买,谨防商家套路,不花一分冤枉钱电视机作为一个家庭里面最主要的家电之一,经过这几年的发展,也发生了很大的变化,人们对电视机的需求从满足型转变成了享受型。因此,对于电视我们在选择的时候,只需要认准三买三不买就可以了