Java源码分析系列笔记-9.CountDownLatch
1. 是什么
不能重复使用的计数器。让一个线程等待其他线程完事再往下执行,类似于Thread.join()
底层使用AQS实现
2. 如何使用
public class CountDownLatchTest
{
public static void main(String[] args) throws InterruptedException
{
CountDownLatch latch = new CountDownLatch(10);
for (int i = 0; i < 10; i++)
{
int finalI = i;
new Thread(() -> {
try
{
if (finalI == 5)
{
TimeUnit.SECONDS.sleep(10L);
}
System.out.println(String.format("线程%s,时间【%s】 countdown", Thread.currentThread().getName(),LocalDateTime.now()));
latch.countDown();
System.out.println(String.format("线程%s,时间【%s】 执行完毕", Thread.currentThread().getName(),LocalDateTime.now()));
}
catch (InterruptedException e)
{
e.printStackTrace();
}
}).start();
}
latch.await();
System.out.println(Thread.currentThread().getName() + "开始执行");
}
}
- 注意
这里countdown的线程不会互相等待,谁先执行完谁就先退出
2.1. CountDownLatch VS CyclicBarrier
| CountDownLatch | CyclicBarrier | |
|---|---|---|
| 使用场景 | 一个线程等待其他线程执行完毕,再往下执行 | 所有线程相互等待直到最后一个线程到达,再往下执行 |
| 能否重复使用 | 不可以 | 可以 |
| 底层实现 | AQS | Lock+Condition |
3. uml

4. 构造方法
public class CountDownLatch {
//继承了AQS
private final Sync sync;
public CountDownLatch(int count) {
if (count < 0) throw new IllegalArgumentException("count < 0");
//默认就设置了count个信号量(即相当于一开始就加锁了count次)
this.sync = new Sync(count);
}
}
4.1. Sync【AQS子类】
private static final class Sync extends AbstractQueuedSynchronizer {
Sync(int count) {
setState(count);
}
int getCount() {
return getState();
}
//重写的是AQS共享获取锁的方法
protected int tryAcquireShared(int acquires) {
return (getState() == 0) ? 1 : -1;
}
//重写的是AQS共享释放锁的方法
protected boolean tryReleaseShared(int releases) {
// Decrement count; signal when transition to zero
for (;;) {
int c = getState();
if (c == 0)
return false;
int nextc = c-1;
if (compareAndSetState(c, nextc))
return nextc == 0;
}
}
}
5. countDown方法
public void countDown() {
//调用了AQS的releaseShared方法
sync.releaseShared(1);
}
5.1. 使用AQS释放锁
- AQS releaseShared
public final boolean releaseShared(int arg) {
//调用Sync重写的tryReleaseShared释放信号量
if (tryReleaseShared(arg)) {
//释放锁成功后调用Sync的doReleaseShared方法
doReleaseShared();
return true;
}
return false;
}
- 3行:调用AQS的tryReleaseShared方法释放锁,由于Sync重写了这个方法,所以调用的是Sync重写的tryReleaseShared释放锁。当锁的数量减为0返回ture,表明所有线程都准备就绪
- 5行:使用tryReleaseShared释放锁成功后调用Sync的doReleaseShared方法。移除AQS队列中SIGNAL的节点并一个个唤醒
下面具体说明:
5.1.1. 尝试释放锁
- Sync.tryReleaseShared
protected boolean tryReleaseShared(int releases) {
// Decrement count; signal when transition to zero
//不断尝试
for (;;) {
//信号量为0,表明还没有人加锁,自然没法解锁,返回失败
int c = getState();
if (c == 0)
return false;
//CAS设置信号量-1。
int nextc = c-1;
if (compareAndSetState(c, nextc))
//看是否为0,是则返回成功
return nextc == 0;
}
}
5.1.2. 所有锁释放成功后,移除AQS队列中SIGNAL的节点,并一个个唤醒
- doReleaseShared
private void doReleaseShared() {
//不断尝试
for (;;) {
Node h = head;
//AQS队列不为空,把队列中SIGNAL的节点移除
if (h != null && h != tail) {
int ws = h.waitStatus;
//头节点状态为SIGNAL
if (ws == Node.SIGNAL) {
//在头节点状态为signal的情况设置为0,失败了继续直到成功
if (!compareAndSetWaitStatus(h, Node.SIGNAL, 0))
continue; // loop to recheck cases
//把头节点从AQS队列中移除
unparkSuccessor(h);
}
//头节点状态为0,那么设置为PROPAGATE,失败了继续直到成功
else if (ws == 0 &&
!compareAndSetWaitStatus(h, 0, Node.PROPAGATE))
continue; // loop on failed CAS
}
//队列中没有SIGNAL的节点
if (h == head) // loop if head changed
break;
}
}
5.1.2.1. 把头节点从AQS队列中移除
- unparkSuccessor
private void unparkSuccessor(Node node) {
int ws = node.waitStatus;
//当前节点的状态<0,则把状态改为0
//0是空的状态,因为node这个节点的线程释放了锁后续不需要做任何
if (ws < 0)
compareAndSetWaitStatus(node, ws, 0);
//当前节点的下一个节点为空或者状态>0(即是取消状态)
Node s = node.next;
if (s == null || s.waitStatus > 0) {
s = null;
//那么从队尾开始往前遍历找到离当前节点最近的下一个状态<=0的节点(即非取消状态)
for (Node t = tail; t != null && t != node; t = t.prev)
if (t.waitStatus <= 0)
s = t;
}
//唤醒下一个节点
if (s != null)
LockSupport.unpark(s.thread);
}
6. await方法
public void await() throws InterruptedException {
//调用AQS的acquireSharedInterruptibly方法加锁
sync.acquireSharedInterruptibly(1);
}
6.1. 使用AQS加锁
- AQS acquireSharedInterruptibly
public final void acquireSharedInterruptibly(int arg)
throws InterruptedException {
if (Thread.interrupted())
throw new InterruptedException();
//调用Sync重写的tryAcquireShared判断是否加锁。
if (tryAcquireShared(arg) < 0)
doAcquireSharedInterruptibly(arg);
}
- 6行:调用Sync重写的tryAcquireShared判断是否需要加锁,而不是真的加锁。可以看出当tryAcquireShared返回<0的时候需要往下执行doAcquireSharedInterruptibly进行加锁。
而tryAcquireShared返回<0指的是一开始设置的count个信号量没有被用完,说明其他线程任务没执行完 - 7行:加锁。准确说是加入AQS队列,阻塞等待其他线程执行完
下面详细说明:
6.1.1. 判断是否需要加锁
- Sync tryAcquireShared
protected int tryAcquireShared(int acquires) {
//当前锁的数量为0,即所有线程任务都执行完了,那么返回1不用加锁
//否则>0指的是一开始设置的count个信号量没有被用完,说明其他线程任务没执行完。那么该线程需要进行加锁,故返回-1
return (getState() == 0) ? 1 : -1;
}
6.1.2. 需要加锁,那么加入AQS队列阻塞等待其他线程执行完
当state>0说明有信号量没被释放完,那么需要加锁
- doAcquireSharedInterruptibly
private void doAcquireSharedInterruptibly(int arg)
throws InterruptedException {
//以SHARE模式加入AQS队列
final Node node = addWaiter(Node.SHARED);
boolean failed = true;
try {
//死循环直到获取锁成功
for (;;) {
//逻辑1.
//当前节点的前一个节点是头节点的时候(公平锁:即我的前面没有人等待获取锁),尝试获取锁
final Node p = node.predecessor();
if (p == head) {
//state == 0(即没人加锁的情况下才执行加锁--其实并没有真的加锁)
int r = tryAcquireShared(arg);
if (r >= 0) {
//获取锁成功后设置头节点为当前节点
setHeadAndPropagate(node, r);
p.next = null; // help GC
failed = false;
return;
}
}
//逻辑2.
//当前节点的前一个节点状态时SIGNAL(承诺唤醒当前节点)的时候,阻塞当前线程。
//什么时候唤醒?释放锁的时候
//唤醒之后干什么?继续死循环执行上面的逻辑1
if (shouldParkAfterFailedAcquire(p, node) &&
parkAndCheckInterrupt())
throw new InterruptedException();
}
//如果发生了异常,那么执行下面的逻辑
} finally {
//除了获取锁成功的情况都会执行cancelAcquire方法
if (failed)
cancelAcquire(node);
}
}
6.1.2.1. 构造节点加入AQS队列
- AQS.addWaiter
private Node addWaiter(Node mode) {
//用当前线程、SHARED模式构造节点
Node node = new Node(Thread.currentThread(), mode);
// 队列不为空
Node pred = tail;
if (pred != null) {
//插入到队尾
node.prev = pred;
if (compareAndSetTail(pred, node)) {
pred.next = node;
return node;
}
}
//队列为空或者插入到队尾失败
enq(node);
return node;
}
队列为空或者插入到队尾失败的情况执行enq,如下
- AQS.enq
private Node enq(final Node node) {
//死循环直到入队成功
for (;;) {
Node t = tail;
//队列为空,那么初始化头节点。注意是new Node而不是当前node(即队头是个占位符)
if (t == null) {
if (compareAndSetHead(new Node()))
tail = head;
//队列不为空,插入到队尾
} else {
node.prev = t;
if (compareAndSetTail(t, node)) {
t.next = node;
return t;
}
}
}
}
6.1.2.2. 判断是否需要阻塞
- shouldParkAfterFailedAcquire
//根据(前一个节点,当前节点)->是否阻塞当前线程
private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {
int ws = pred.waitStatus;
//前一个节点的状态时SIGNAL,即释放锁后承诺唤醒当前节点,那么返回true可以阻塞当前线程
if (ws == Node.SIGNAL)
return true;
//前一个节点状态>0,即CANCEL。
//那么往前遍历找到没有取消的前置节点。同时从链表中移除CANCEL状态的节点
if (ws > 0) {
do {
node.prev = pred = pred.prev;
} while (pred.waitStatus > 0);
pred.next = node;
// 前置节点状态>=0,即0或者propagate。
//这里通过CAS把前置节点状态改成signal成功获取锁,失败的话再阻塞。why?
} else {
compareAndSetWaitStatus(pred, ws, Node.SIGNAL);
}
return false;
}
6.1.2.3. 真正阻塞
- parkAndCheckInterrupt
private final boolean parkAndCheckInterrupt() {
//使用Unsafe阻塞当前线程,这里会清除线程中断的标记,因此需要返回中断的标记
LockSupport.park(this);
return Thread.interrupted();
}
6.1.3. 不需要加锁
当state=0说明所有信号量已被释放完,那么直接返回,执行业务逻辑
7. 总结
- 让一个线程等待其他线程完事再往下执行,类似于Thread.join()
- 主线程创建CountDownLatch的时候初始化了信号量,相当于一开始就有N个人加锁。
- 主线程调用await的时候检查信号量是否为0,不为0说明其他线程没有执行完,那么加入AQS队列阻塞,等待唤醒
- 其他线程调用countDown的时候会使信号量-1,最后一个线程减为0的时候会唤醒AQS队列中的所有节点(主线程),让其继续往下执行
- 主线程被唤醒继续往下执行
8. 参考
Java源码分析系列笔记-9.CountDownLatch的更多相关文章
- Java源码分析系列之HttpServletRequest源码分析
从源码当中 我们可以 得知,HttpServletRequest其实 实际上 并 不是一个类,它只是一个标准,一个 接口而已,它的 父类是ServletRequest. 认证方式 public int ...
- Java源码分析系列
1) 深入Java集合学习系列:HashMap的实现原理 2) 深入Java集合学习系列:LinkedHashMap的实现原理 3) 深入Java集合学习系列:HashSet的实现原理 4) 深入Ja ...
- MyCat源码分析系列之——结果合并
更多MyCat源码分析,请戳MyCat源码分析系列 结果合并 在SQL下发流程和前后端验证流程中介绍过,通过用户验证的后端连接绑定的NIOHandler是MySQLConnectionHandler实 ...
- MyCat源码分析系列之——BufferPool与缓存机制
更多MyCat源码分析,请戳MyCat源码分析系列 BufferPool MyCat的缓冲区采用的是java.nio.ByteBuffer,由BufferPool类统一管理,相关的设置在SystemC ...
- [Tomcat 源码分析系列] (二) : Tomcat 启动脚本-catalina.bat
概述 Tomcat 的三个最重要的启动脚本: startup.bat catalina.bat setclasspath.bat 上一篇咱们分析了 startup.bat 脚本 这一篇咱们来分析 ca ...
- MyBatis 源码分析系列文章导读
1.本文速览 本篇文章是我为接下来的 MyBatis 源码分析系列文章写的一个导读文章.本篇文章从 MyBatis 是什么(what),为什么要使用(why),以及如何使用(how)等三个角度进行了说 ...
- spring源码分析系列 (8) FactoryBean工厂类机制
更多文章点击--spring源码分析系列 1.FactoryBean设计目的以及使用 2.FactoryBean工厂类机制运行机制分析 1.FactoryBean设计目的以及使用 FactoryBea ...
- spring源码分析系列 (5) spring BeanFactoryPostProcessor拓展类PropertyPlaceholderConfigurer、PropertySourcesPlaceholderConfigurer解析
更多文章点击--spring源码分析系列 主要分析内容: 1.拓展类简述: 拓展类使用demo和自定义替换符号 2.继承图UML解析和源码分析 (源码基于spring 5.1.3.RELEASE分析) ...
- spring源码分析系列 (1) spring拓展接口BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessor
更多文章点击--spring源码分析系列 主要分析内容: 一.BeanFactoryPostProcessor.BeanDefinitionRegistryPostProcessor简述与demo示例 ...
- spring源码分析系列 (3) spring拓展接口InstantiationAwareBeanPostProcessor
更多文章点击--spring源码分析系列 主要分析内容: 一.InstantiationAwareBeanPostProcessor简述与demo示例 二.InstantiationAwareBean ...
随机推荐
- HarmonyOS NEXT 基于原生能力获取视频缩略图
大家好,我是 V 哥. 不得不佩服 HarmonyOS NEXT 原生能力的强大,如果你想在 鸿蒙 APP 开发中获取视频缩略图,不用依赖第三方库,就可以高效和稳定的实现,AVMetadataHelp ...
- 探秘Transformer系列之(19)----FlashAttention V2 及升级版本
探秘Transformer系列之(19)----FlashAttention V2 及升级版本 目录 探秘Transformer系列之(19)----FlashAttention V2 及升级版本 0 ...
- 【Spring】JdbcTemplate的使用方法
概念和准备 什么是 JdbcTemplate Spring 框架对 JDBC 进行封装,使用 JdbcTemplate 方便实现对数据库操作 准备工作 引入相关 jar 包 在 spring 配置文件 ...
- 重磅!微信官方恢复了个人红包封面的制作入口,限时开放!!.md
前两天微信开放了个人红包封面,引起了大家欢呼雀跃~ 可惜--没几个小时,因为一个不可描述的原因,官方小程序下架了-- 但是, 现在好消息来啦, 官方又恢复个人红包封面制作啦~ 本文教你如何制作红包封面 ...
- ST表 RMQ问题(区间最大/最小值查询)
#include <iostream> #include <cstring> #include <algorithm> #include <cmath> ...
- redis的fd与epoll是怎么使用的
Redis 的高性能网络模型核心依赖于 文件描述符(fd) 和 epoll 的协同工作.下面我将从底层机制到实际应用,详细解析它们的配合方式: 一.核心组件关系图 二.fd 在 Redis 中的具体应 ...
- ThreadPoolExecutor的内部类Worker详细解析
一.定义 ThreadPoolExecutor 的内部类 Worker 是线程池的核心实现之一,它封装了线程和任务,并负责执行任务.Worker 类继承自 AbstractQueuedSynchron ...
- Redis 持久化——混合持久化
1.Redis 持久化--混合持久化 RDB 和 AOF 持久化各有利弊,RDB 可能会导致一定时间内的数据丢失,而 AOF 由于文件较大则会影响 Redis 的启动速度,为了能同时使用 RDB 和 ...
- 🎀gh-ost工具介绍及使用
简介 gh-ost 是一款由GitHub开发的在线DDL(Online Data Definition Language)变更工具,专门用于MySQL数据库.它允许在不锁定表的情况下执行数据库模式变更 ...
- MySQL 的查询优化器如何选择执行计划?
MySQL 的查询优化器负责决定如何执行 SQL 查询,它会根据多个因素选择最优的执行计划.查询优化器的目标是选择一个成本最低.性能最优的执行计划,以便高效地处理查询.执行计划的选择是基于 MySQL ...