用“老司机发车”的案例带你详细了解 CountDownLatch 的执行流程
创始人
2025-05-31 16:33:26

哈喽大家好,我是阿Q。

前几天我们把 ReentrantLock的原理 进行了详细的讲解,不熟悉的同学可以翻看前文,今天我们介绍另一种基于 AQS 的同步工具——CountDownLatch。

CountDownLatch 被称为倒计时器,也叫闭锁,是 juc 包下的工具类,同时也是共享锁的一种实现。它的作用是可以让一个或多个线程等待,直到所有线程的任务都执行完之后再继续往下执行。

举个简单的例子:阿Q高中时期都是乘坐大巴往返于县城与农村,那时的司机为了利益的最大化,会在汽车满员的情况下才会发车。

如果我们把乘客去车站乘车比作一个一个的线程,那 CountDownLatch 做的事就是等大家到齐之前的等待工作。

我们从源码的角度来分析下它的工作原理

①谁来决定公交车上的座位数?

公交车上的座位数是由汽车制造商决定的,在 CountDownLatch 中也会存在这样一个值 count,用来表示需要等待的线程个数

count 值是在 CountDownLatch 的构造函数中进行初始化的

public CountDownLatch(int count) {if (count < 0) throw new IllegalArgumentException("count < 0");this.sync = new Sync(count);
}Sync(int count) {//设置 AQS 中的 state 为 count 值setState(count);
}

计数值 count 是一次性的,当它的值减为0后就不会再变化了,这也是其存在的不足之处。

②谁来确定乘客全部到齐?

在汽车发车前检票员会对车上的乘客数量进行清点,如果满员了就会通知司机开车。

当然也可以采用这种方法:在得知车座位数的前提下,每上来一位乘客,座位数进行减一操作。CountDownLatch 就是采用的上述方法,它的 countDown() 方法会对 state 的值执行减1操作。

让我们从源码的角度来认识一下该方法。

public void countDown() {//释放共享锁sync.releaseShared(1);
}public final boolean releaseShared(int arg) {if (tryReleaseShared(arg)) {doReleaseShared();return true;}return false;
}

先尝试释放锁,如果返回 true,则执行释放操作,反之不执行。我们分析下上边的两个方法

protected boolean tryReleaseShared(int releases) {for (;;) {//获取当前等待的线程数量int c = getState();//等待线程数为0,表示没有等待线程,故不需要释放锁资源if (c == 0)return false;//执行减1操作int nextc = c-1;//自旋+CAS将state的属性值-1if (compareAndSetState(c, nextc))return nextc == 0;}
}

最后一步中,如果减一之后为0,则说明没有其它线程等待,需要执行释放锁操作,返回 true,反之不需要。

在开始分析 doReleaseShared() 之前,我们先来补全一下 AQS 中 waitStatus 的状态说明

  • 初始化状态:0,表示当前节点在同步队列中,等待获取锁;
  • CANCELLED:1,表示当前节点取消获取锁;
  • SIGNAL:-1,表示后续节点等待当前节点唤醒;
  • CONDITION:-2,表示当前线程正在条件等待队列中;
  • PROPAGATE:-3,共享模式,前置节点唤醒后续节点后,唤醒操作无条件传播下去;
/*** 释放锁:唤醒后续节点*/
private void doReleaseShared() {for (;;) {Node h = head;//不是null 且不为尾节点,因为尾节点没有后续节点需要唤醒了if (h != null && h != tail) {int ws = h.waitStatus;//只有状态为 -1 才可以唤醒后续节点if (ws == Node.SIGNAL) {//将waitStatus设置为0失败会继续循环if (!compareAndSetWaitStatus(h, Node.SIGNAL, 0))continue;unparkSuccessor(h);}//将waitStatus设置为PROPAGATE失败会继续循环else if (ws == 0 &&!compareAndSetWaitStatus(h, 0, Node.PROPAGATE))continue;                }if (h == head)                   break;}
}

unparkSuccessor() 方法用于唤醒 AQS 中被挂起的线程,在ReentrantLock的原理中讲过了,此处不再赘述。

小结:当线程使用 countDown() 方法时,其实是使用了 tryReleaseShared() 方法以 CAS 的操作来减少 state ,直至 state 为 0 ,进而释放锁资源,唤醒后续节点。

③谁来发车?

肯定是司机来发车呀,那我们的 CountDownLatch 是如何实现的呢?

CountDownLatch 中的 await() 方法,就是等待线程的总开关,当发现 state 的值为0时会释放所有的等待线程,发车了。

我们从源码角度来看下它是如何工作的

public void await() throws InterruptedException {sync.acquireSharedInterruptibly(1);
}public final void acquireSharedInterruptibly(int arg)throws InterruptedException {//如果线程中断了,直接抛出中断异常if (Thread.interrupted())throw new InterruptedException();//如果小于0,代表 state 不为0,即还有任务未执行完毕,会执行获取共享锁的操作if (tryAcquireShared(arg) < 0)doAcquireSharedInterruptibly(arg);
}protected int tryAcquireShared(int acquires) {return (getState() == 0) ? 1 : -1;
}

我们来看看它到底是如何获取共享锁的

private void doAcquireSharedInterruptibly(int arg)throws InterruptedException {//将当前线程封装成node放到队尾final Node node = addWaiter(Node.SHARED);boolean failed = true;try {for (;;) {final Node p = node.predecessor();if (p == head) {int r = tryAcquireShared(arg);//state为0,表示此时等待线程全部执行完毕,r为1。if (r >= 0) {setHeadAndPropagate(node, r);p.next = null;failed = false;return;}}//从当前node节点向前寻找有效节点,并保证有效节点的waitStatus状态为-1if (shouldParkAfterFailedAcquire(p, node) &&//挂起线程parkAndCheckInterrupt())//在拿锁的期间,如果被中断了,那么会抛出异常,取消拿锁throw new InterruptedException();}} finally {if (failed)//将当前节点设置为失效节点,并挂到最近的有效节点后边,上文中有图解cancelAcquire(node);}
}

其中最重要的就是 setHeadAndPropagate() 方法

private void setHeadAndPropagate(Node node, int propagate) {Node h = head; //将当前node设置为head,并将node的线程置为空setHead(node);if (propagate > 0 || h == null || h.waitStatus < 0 ||(h = head) == null || h.waitStatus < 0) {Node s = node.next;if (s == null || s.isShared())//释放锁:唤醒后续节点doReleaseShared();}
}

小结:当线程使用 await() 方法时会将当前线程封装成 node 加入AQS 队列中,如果发现 state 不为0,说明还有任务未执行完成,继续阻塞;如果 state 为0,会释放掉所有的等待线程,执行 await() 之后的数据。

流程图了解一下

理论讲完了,那我们用代码来演示下上边的例子

public static void main(String[] args) throws InterruptedException {int count = 10;//设置线程池并发数ExecutorService executorService = Executors.newFixedThreadPool(count);//假设大巴可以拉十个乘客,初始化stateCountDownLatch countDownLatch = new CountDownLatch(count);for (int i = 0; i < count; i++) {final int num = i;executorService.execute(()->{try {Thread.sleep((long) (new Random().nextDouble() * 3000) + 1000);System.out.println("乘客坐在了"+ (num +1) + "号座位上");} catch (InterruptedException exception) {exception.printStackTrace();}finally {countDownLatch.countDown();}});}System.out.println("司机等待乘客上车");countDownLatch.await();System.out.println("发车了");executorService.shutdown();
}

执行结果如下:

细心地同学肯定会问了:如果遇上刮风下雨,来坐车的人少了,那已经上车的乘客岂不是回不了家了?

当然不是了,大巴其实也是有时间观念的,即使车上的乘客不满员到了一定的时间司机也会发车的,另外还会在路上顺道稍几个人上车。那我们的 CountDownLatch 是如何实现的呢?

CountDownLatch 还提供了一个 await(long timeout, TimeUnit unit)方法,在一定的时间间隔内会阻塞当前线程,等待 count 个线程执行任务,一旦超出了等待时间,便会继续往下执行。

我们将上边的countDownLatch.await();替换为countDownLatch.await(3, TimeUnit.SECONDS);,执行结果如下所示

上文中的例子是 CountDownLatch 的其中一种用法,即主线程等待其他线程执行完毕之后再执行。它还有另一种用法,即实现多个线程开始执行任务的最大并行性,类似发令枪响前,运动员统一在起跑线就位的场景。

public static void main(String[] args) throws InterruptedException {//设置线程池并发数ExecutorService executorService = Executors.newFixedThreadPool(10);CountDownLatch countDownLatch = new CountDownLatch(1);//一组有6名运动员for (int i = 0; i < 6; i++) {final int num = i;executorService.execute(()->{try {System.out.println("运动员"+ (num+1) +"等待发令枪响");countDownLatch.await();System.out.println("运动员"+ (num+1) +"开始起跑");} catch (InterruptedException exception) {exception.printStackTrace();}});}Thread.sleep(3000);countDownLatch.countDown();System.out.println("发令枪响");executorService.shutdown();
}

执行结果如下

说了这么多,都是样例?你有没有在项目中应用过呢?

回答当然是“Yes”了,之前的运营端有个统计页面,要求统计用户新增数量、订单数量、商品交易总额等多张表的指标值,为了提高执行速率,我就启用了多个子线程分别去统计,用 CountDownLatch 来等待它们的统计结果。

今天的内容到这里就结束了,希望对大家有所帮助,我们下期再见。

跪求一键三连,更文很累的,不要白嫖我,需要一点正反馈。点击名片与我联系,希望在这个冷漠的城市里,让我们互相温暖。

推荐阅读

实战:画了几张图,终于把OAuth2搞清楚了

重磅出击,20张图带你彻底了解ReentrantLock加锁解锁的原理

领导看了我写的关闭超时订单,让我出门左转!

看了同事写的代码,我竟然开始默默的模仿了。。。

实战篇:断点续传?文件秒传?手撸大文件上传

相关内容

热门资讯

城市住房产品力变化 | 探索“... 记者 田国宝 编者按:推开一扇门,有人看见三代同堂的烟火,有人撞见独自拼凑的晨昏,有人在精装样板间描...
破发股爱科赛博5高管拟减持 2... 中国经济网北京6月6日讯 爱科赛博(688719.SH)昨晚发布董事、高级管理人员减持股份计划公告。...
2025年前5月私募证券基金备... 2025年私募证券基金市场展现出强劲复苏势头。前5个月全行业累计备案私募证券基金4361只,较去年同...
始于雷军、困于罗永浩、终于董明... 在孟羽童和董明珠“世纪和解”重回格力直播后,沉寂了近一年的王自如在社交平台露面了。 6月5日晚,王自...
霸王茶姬首份成绩单背后,如何从... 如今的新茶饮市场高度内卷,价格战频发,霸王茶姬凭什么能做到稳健增长?文|《中国企业家》记者胡楠楠编辑...
金陵饭店2024年净利润下滑4... 本报记者 李贝贝 上海报道2024年,金陵饭店股份有限公司(下称“金陵饭店”,601007.SH)归...
3年卖掉69座电站!林洋能源再... 本报(chinatimes.net.cn)记者李佳佳 李未来 北京报道近期,江苏林洋能源股份有限公司...
高考考点:二桃杀三士 如何应对... 祝各位考生高考节快乐。上次讲崔杼弑其君,星空君故意漏了一个细节。齐庄公死后,著名的退居二线的大夫晏婴...
非农数据提振市场,美国三大股指... 美股周五收高,道指上涨逾400点,标普500指数突破6000点整数关口。三大股指本周均录得涨幅。Wi...
美股上攻,黄金下跌超1% 美股... 2025.06.07本文字数:2393,阅读时长大约4分钟作者 |第一财经 樊志菁*三大股指涨超1%...
加力支持 精准滴灌 多方协同—...   新华社北京6月6日电 题:加力支持 精准滴灌 多方协同——金融护航外贸发展观察   新华社记者姚...
卖出600万只枕头的亚朵,栽在... 「核心提示」一只枕套,将加盟商和亚朵之间利益共生又充满矛盾的关系摆到了台面上。作者 |詹方歌邢昀一年...
“天价耳环事件”最新消息!“黄... 近期,因“天价耳环”事件备受关注的黄杨钿甜及其父亲杨伟持续引发热议。有网友发现其父杨伟在公开回应前,...
年轻人涌入医院买面包?为啥医院... 说起最近几年,各种消费热点中有一类非常特殊,这就是医院成为了不少消费的热点,各种中药咖啡、中药饮料、...
金穗春操纵股票遭证监会罚没过亿... 6月6日晚间,证监会披露的一则行政处罚决定书(2025 59号)显示,金穗春(男)操纵股票被证监会罚...
证监会核准中央汇金成为8家公司... 南财早新闻,早听早知道。1、上交所召开高分红重回报暨上市公司价值提升座谈会。上交所相关负责人表示,未...
信托业破局价格“内卷”:少赚吆... 1万元就可以做一单资产证券化业务——信托本源业务的低价“内卷”,正困扰着信托公司。 随着信托业转型加...
每周股票复盘:爱博医疗(688... 截至2025年6月6日收盘,爱博医疗(688050)报收于71.05元,较上周的71.41元下跌0....
大转向!外资,爆买! 大转向如... 外资突然出现重大转变。在连续四个月净卖出后,外资在5月大举押注亚洲股市。据伦敦证券交易所集团(LSE...
不想对话!特朗普余气未消,特斯... 在双方爆发激烈“口水战”后,美国总统特朗普与特斯拉首席执行官(CEO)马斯克并未如外界期望的那样在周...
惊天大骗局,一万人踩雷 惊天大... 一万多人辛苦一辈子的血汗钱瞬间化为乌有。这再次说明了,天上不会掉馅饼。所有的投资都要基于常识,正如中...
广州农商银行60后副行长再担重... 担任广州农商银行副行长三年多后,李亚光再添职务,其首席信息官的任职资格获批。湘财Plus注意到,李亚...
标普500收复6000点!美股... *三大股指涨超1%;*加拿大产区大火持续,原油延续反弹;*特斯拉反弹超3%,市场关注特朗普与马斯克关...
诺诚健华涨2.42%,成交额1... 6月6日,诺诚健华涨2.42%,成交额1.28亿元,换手率2.01%,总市值432.89亿元。 异动...
远光软件跌0.34%,成交额1... 6月6日,远光软件跌0.34%,成交额1.72亿元,换手率1.72%,总市值111.83亿元。 异动...
非农数据提振市场,美国三大股指...   中新经纬6月7日电 美股周五收高,道指上涨逾400点,标普500指数突破6000点整数关口。三大...
每周股票复盘:招商银行(600... 截至2025年6月6日收盘,招商银行(600036)报收于44.47元,较上周的43.43元上涨2....
小阔集团董事长尹阔:民营经济促... 中国经济网北京6月6日讯(记者 李方)“民营经济促进法犹如一颗‘定心丸’,为我们营造了一个稳定、公平...
Circle纽交所上市:大涨1... 雷递网 雷建平 6月6日 USDC稳定币发行商Circle(股票代码为:“CRCL”)昨日在纽交所上...
招商银行成功发行50亿绿色金融... 2025年6月5日,在第54个“世界环境日”到来之际,招商银行在全国银行间债券市场成功簿记发行绿色金...