深入理解CountDownLatch

news2025/1/12 22:48:36

CountDownLatch是一个同步协助类,通常用于一个或多个线程等待,直到其他线程完成某项工作。

CountDownLatch使用一个计数值进行初始化,调用它提供的await()方法的线程会被阻塞直到该计数值减为0。减计数值的方法是countDown(),该方法可以在同一个线程中多次调用,也可以在多个线程中被调用,当计数值减为0时所有调用await()方法的线程被唤醒。

API

CountDownLatch的构造函数定义如下,count的值被赋值给state。

public CountDownLatch(int count) {
    if (count < 0) throw new IllegalArgumentException("count < 0");
    this.sync = new Sync(count);
}

CountDownLatch的常用方法如下:

//当前线程阻塞,直到count/state的值变为0
public void await() throws InterruptedException {
    sync.acquireSharedInterruptibly(1);
}
//当前线程阻塞,直到count/state的值变为0或等待timeout的时间
public boolean await(long timeout, TimeUnit unit) throws InterruptedException {
    return sync.tryAcquireSharedNanos(1, unit.toNanos(timeout));
}
//将count/state的值减1
public void countDown() {
    sync.releaseShared(1);
}

使用场景

CountDownLatch一般可以用于两个场景:

  1. 多个线程等待,随后并发执行;
  2. 单个线程等待,汇总合并多个线程的执行结果。

多个线程等待

这种场景下,通常是多个线程调用await()方法阻塞,直到其他线程调用countDownLatch()将count的值减为0,如下:

public static void main(String[] args) throws InterruptedException {
    CountDownLatch countDownLatch = new CountDownLatch(1);
    for (int i = 0; i < 5; i++) {
        new Thread(() -> {
            try {
                //准备完毕……运动员都阻塞在这,等待号令
                countDownLatch.await();
                String parter = "【" + Thread.currentThread().getName() + "】";
                System.out.println(parter + "开始执行……");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }).start();
    }
    Thread.sleep(2000);// 裁判准备发令
    countDownLatch.countDown();// 发令枪:执行发令
}

单个线程等待

在很多场景中,主流程需要等待多个不同的任务完成后再处理结果,此时就要求主流程线程需要阻塞直到所有任务执行完成,如下:

public class CountDownLatchTest {

    private static volatile int count = 0;

    public static void main(String[] args) throws Exception {

        CountDownLatch countDownLatch = new CountDownLatch(5);
        for (int i = 0; i < 5; i++) {
            final int index = i;
            new Thread(() -> {
                try {
                    Thread.sleep(1000 + ThreadLocalRandom.current().nextInt(1000));
                    count++;
                    System.out.println(Thread.currentThread().getName()+" finish task" + index );

                    countDownLatch.countDown();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }).start();
        }

        // 主线程在阻塞,当计数器==0,就唤醒主线程往下执行。
        countDownLatch.await();
        System.out.println("所有任务执行完成,count=" + count);

    }
}

源码解析

这里主要介绍最常用的await()和countDown()方法的源码,以及CountDownLatch的原理。

await()

  1. await()
public void await() throws InterruptedException {
    sync.acquireSharedInterruptibly(1);
}
  1. acquireSharedInterruptibly(permits)方法是AQS中定义的共享锁获取锁的通用方法,实现如下:
public final void acquireSharedInterruptibly(int arg) throws InterruptedException {
	if (Thread.interrupted())
        throw new InterruptedException();
    if (tryAcquireShared(arg) < 0)
        doAcquireSharedInterruptibly(arg);
}
  1. tryAcquireShared(arg)是提供给子类实现的模版方法,该方法在CountDownLatch中的实现如下:
protected int tryAcquireShared(int acquires) {
    //如果state=0,则返回正值1,否则返回负值-1
    return (getState() == 0) ? 1 : -1;
}
  1. 在第2步中,tryAcquireShared(arg)返回值小于0则调用doAcquireSharedInterruptibly(arg)方法阻塞当前线程,如果返回值大于等于0则方法执行结束,而在第3步的tryAcquireShared(arg)实现可以看出来,当state!=0时调用await()方法的线程都会阻塞,只有state=0时才不会阻塞。下面的测试程序和执行结果可以验证以上结论:
public static void main(String[] args) throws InterruptedException {
    CountDownLatch latch = new CountDownLatch(1);
    new Thread(() -> {
        try {
            System.out.println(Thread.currentThread().getName() + ":我阻塞了...");
            latch.await();
            System.out.println(Thread.currentThread().getName() + ":我被唤醒了...");
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
    },"thread-branch").start();
    Thread.sleep(100);
    System.out.println("调用countDown()方法前,count=" + latch.getCount());
    latch.countDown();
    Thread.sleep(100);
    System.out.println("调用countDown()方法后,count=" + latch.getCount());
    latch.await();
    System.out.println("这里如果执行了,说明主线程没有阻塞,count=" + latch.getCount());
}

  1. doAcquireSharedInterruptibly(arg)方法是AQS中的共享锁入队同步等待队列并阻塞通用方法,在Semaphore中详细介绍过,此处不再赘述。

countDown()

countDown()方法的主要作用是将state减1,当state=0时则需要唤醒所有在同步等待队列中阻塞的线程。

  1. countDown()实现如下:
public void countDown() {
    sync.releaseShared(1);
}
  1. releaseShared()方法是AQS中提供的释放共享锁的通用方法,实现如下:
public final boolean releaseShared(int arg) {
    if (tryReleaseShared(arg)) {
        doReleaseShared();
        return true;
    }
    return false;
}
  1. tryReleaseShared()方法是AQS定义的模版方法,在CounDownLatch中的实现如下:
protected boolean tryReleaseShared(int releases) {
    for (;;) {
        int c = getState();
        //判断如果state=0,则不需要再唤醒同步等待队列了,因为之前已经唤醒过了
        if (c == 0)
            return false;
        //这里就是countDown()方法的主要作用,将state-1
        int nextc = c-1;
        //如果使用CAS修改state-1失败,则循环修改直到成功
        if (compareAndSetState(c, nextc))
            //修改state成功后,如果state=0,则返回true
            return nextc == 0;
    }
}
  1. 在第3步中,只有当前这次执行countDown()方法将state的值减为0后,才会返回true,此时才会去执行第2步中的doReleaseShared()方法,该方法将唤醒同步等待队列所有阻塞线程。

CountDownLatch的使用流程可以总结如下:

  1. 使用一个不小于0的整数初始化CountDownLatch,该整数值赋给state;
  2. 在state不为0时,所有调用await()方法的线程都会进入同步等待队列阻塞;当state=0后,再调用await()方法不会做任何操作;
  3. 每次调用countDown()方法时,先判断state是否已经是0了,如果是0则什么都不做直接返回,如果不是0则将state-1后使用CAS更新state;
  4. 如果更新失败,则循环调用CAS更新直到更新成功;
  5. 更新成功后,判断state是否为0,如果不是则什么都不做,此时countDown()方法只是将state-1;如果state=0,则需要唤醒同步等待队列的所有阻塞线程。至此,CountDownLatch执行完成。

CountDownLatch和Semaphore

从上面可以看出CountDownLatch和Semaphore都是通过AQS的共享锁实现的,虽然它们的实现效果截然不同,但是比较它们的不同可以帮助我们记忆它们各自的实现。

  1. Semaphore在调用acquire()方法获取许可证时将state-1,如果state=0则进入同步等待队列阻塞;CountDownLatch在调用await()方法时,只要state!=0,线程都会进入同步等待队列阻塞;
  2. Semaphore在调用release()方法时无论当前线程是否获取过许可证,许可证state的值都会+1(甚至可以突破初始化时给的值),并调用doReleaseShared()方法唤醒同步等待队列首节点,如果许可证足够则会一直向后唤醒;CountDownLatch在调用countDown()方法时,如果state==0则什么都不做,如果state-1!=0则只更新state,如果state-1==0则会唤醒同步等待队列所有阻塞线程。

以上就决定了Semaphore的作用是限流,CountDownLatch的作用是协助线程同步执行。

需要注意的是,CountDownLatch只能使用一次,即当调用countDown()将state减为0后,当前CountDownLatch对象就没用了。如果想要达到重复使用的目的,可以选择另一个功能较CountDownLatch更强大的CyclicBarrier。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/908539.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

OS 多进程图像

schedule()函数为切换进程函数&#xff0c;pCur.state被置为阻塞态&#xff0c;放入磁盘等待队列 pCur和pNew都为对应pcb&#xff08;为了使参与并发执行的每个程序&#xff0c;包含数据都能独立地运行&#xff0c;在操作系统中必须为之配置一个专门的数据结构&#xff0c;称为…

分布式核心知识

文章目录 前言一、分布式中的远程调用1.1RESTful接口1.2RPC协议1.3区别与联系 二、分布式中的CAP原理 前言 关于分布式核心知识详解 一、分布式中的远程调用 在微服务架构中&#xff0c;通常存在多个服务之间的远程调用的需求。远程调用通常包含两个部分&#xff1a;序列化和通…

【日常记录】win10打印机打印不出来,打印队列里有文档无法删除

Q1&#xff1a;win10打印机打印不出来&#xff0c;打印队列里有文档无法删除 可能原因一&#xff1a;打印机队列有文档卡住了&#xff0c;删除掉就好 打开打印机队列&#xff0c;开始 → 设置 → 设备 → 打印机和扫描仪 → 打开打印机队列 → 暂停并取消队列里所有文档 → 点…

unity开发笔记#230821-手搓一个虚拟摇杆

做unity游戏时&#xff0c;如果是做手机端则需要加一个虚拟摇杆以便玩家控制角色移动&#xff08;做pc端则直接获取键值即可较方便&#xff09;。原理就是用Image制作一个大圆圈住一个Image小圆&#xff0c;玩家拖拽小圆控制角色移动。中心思想是&#xff0c;以小圆中心为(0,0)…

C++day1(笔记整理)

一、Xmind整理&#xff1a; 二、上课笔记整理&#xff1a; 1.第一个c程序&#xff1a;hello world #include <iostream> //#:预处理标识符 //<iostream>:输入输出流类所在的头文件 //istream:输入流类 //ostream:输出流类using namespace std; //std&#x…

*args无疑是解决函数重载的一大创新利器--我用可变数量参数解决了函数重载问题

需求分析 最近遇到这样一个需求&#xff1a;根据用户传递的不同参数数量执行不同的功能。我这几天一直在思考这个问题&#xff1a;如何根据参数数量去执行不同的功能&#xff0c;最初的设想是把不需要的参数设置为NONE或者""再或者" "(后两者引号均表示传空…

Go:测试框架GoConvey 简介

快速开始 GoConvey是一个完全兼容官方Go Test的测试框架&#xff0c;一般来说这种第三方库都比官方的功能要强大、更加易于使用、开发效率更高&#xff0c;闲话少说&#xff0c;先看一个example&#xff1a; package utils import (. "github.com/smartystreets/goconvey…

怎么查看小程序中的会员信息

商家通过查看会员信息&#xff0c;可以更好地了解用户&#xff0c;并为他们提供更个性化的服务和推荐。接下来&#xff0c;就将介绍如何查看会员信息。 商家在管理员后台->会员管理处&#xff0c;可以查看到会员列表。支持搜索会员的卡号、手机号和等级。还支持批量删除会员…

Java原子类

是什么 对多线程访问同一个变量&#xff0c;为了保证线程安全需要加锁&#xff0c;而锁是比较消耗性能的。Java从JDK 1.5开始提供了java.util.concurrent.atomic包&#xff0c;这个包中的原子操作类提供了一种用法简单、性能高效、线程安全地更新一个变量的方式。 jdk1.8新增…

【Python机器学习】实验15 将Lenet5应用于Cifar10数据集(PyTorch实现)

文章目录 CIFAR10数据集介绍1. 数据的下载2.修改模型与前面的参数设置保持一致3. 新建模型4. 从数据集中分批量读取数据5. 定义损失函数6. 定义优化器7. 开始训练8.测试模型 9. 手写体图片的可视化10. 多幅图片的可视化 思考题11. 读取测试集的图片预测值&#xff08;神经网络的…

Linux系统编程:进程信号的保存和阻塞

目录 一. 信号保存和阻塞的相关概念 二. 进程信号的表示 2.1 进程信号在内核中的表示 2.2 sigset_t 类型 三. 信号集操作相关函数 3.1 sigset_t 类型参数设置相关函数 3.2 sigprocmask 函数 3.3 sigpending 函数 四. 演示代码 4.1 将所有信号的处理方式都注册为不退出…

详解Spring的循环依赖问题、三级缓存解决方案源码分析

0、基础&#xff1a;Bean的生命周期 在Spring中&#xff0c;由于IOC的控制反转&#xff0c;创建对象不再是简单的new出来&#xff0c;而是交给Spring去创建&#xff0c;会经历一系列Bean的生命周期才创建出相应的对象。而循环依赖问题也是由Bean的生命周期过程导致的问题&#…

wustojc3010快速求和

#include <stdio.h> int main() {int n;double s;s0;scanf("%d",&n);for(int i1;i<n;i){ss(double)1.0/(i*(i1.0));//强转一下类型}printf("%.5lf",s);return 0;}

Docker私有仓库创建

1.Docker私有仓库搭建 拉取私有仓库镜像并启动私有仓库容器。 访问私有仓库容器&#xff0c;表明私有仓库搭建成功。 此时私有仓库就已经搭建完成了。 2.将本机的镜像传到私有仓库 3.将私有仓库镜像拉取到本地

论文导读|European Journal of Operational Research近期文章精选:旅行商问题专题

推文作者&#xff1a;王松阁 编者按 在“European Journal of Operational Research近期论文精选”中&#xff0c;我们有主题、有针对性地选择了European Journal of Operational Research中一些有趣的文章&#xff0c;不仅对文章的内容进行了概括与点评&#xff0c;而且也对文…

DHCP协议原理与应用

DHCP协议原理与应用 一、DHCP协议概述1.1、场景描述1.1.1、场景描述11.1.2、场景描述21.1.3、场景描述3 二、DHCP协议工作原理2.1、DHCP简介2.2、DHCP协议名词解释2.3、DHCP服务器配置2.4、PC的DHCP设置2.5、DHCP协议工作过程2.6、DHCP协议报文及用途2.7、DHCP报文介绍2.7.1、D…

面试之快速学习STL-迭代适配器

先放一张大图 参考&#xff1a;http://c.biancheng.net/view/7255.html 1. 反向迭代器 例子&#xff1a; std::list<int> values{1,2,3,4,5};auto start_it values.rbegin();const auto end_it values.rend();//start_it end_it std::reverse_iterator<std::lis…

HCIP 三层架构实验

三层架构实验 拓扑和思路拓扑思路LSW配置LSW1LSW2LSW3 DHCPLSW2LSW1 ACL外网冗余 拓扑和思路 拓扑 思路 首先划分网段&#xff0c;然后LSW1和LSW2和R1可以用ospf宣告就行&#xff0c;然后R1写条缺省指向R2 然后可以将LSW1和LSW2三合一&#xff0c;给交换机配置换分组&#x…

用电脑软件0代码设计WS2812显示效果(含软件下载地址)

用电脑软件设计WS2812显示效果 ws2812显示效果设计软件和单片机程序文件 单片机型号为8脚的STC8G1K08A或STC8G1K17A或者16脚的STC8G1K08或STC8G1K17 烧录时晶振选择22.1184M 百度网盘下载地址&#xff1a;链接: https://pan.baidu.com/s/1cVvA604IKtZ-cIqTX8Jgzw?pwd1234 提取…

数学分析:体形式

确实&#xff0c;面积应该是没有正负的&#xff0c;或者说和曲面的定向应该是无关的。我们用微分形式的积分定义了具有参数形式的曲面的面积。所以这个意思就是说&#xff0c;对于不同的曲面的定向&#xff0c;微分形式应该也不同。 这就是体形式的具体样子&#xff0c;得到每…