并发之CountDownLatch用法详解 - 五色风车

yrjns 2021-08-06 原文


并发之CountDownLatch用法详解

概念

       CountDownLatch 是一个同步工具类,它允许一个或多个线程一直等待,直到其他线程执行完后再执行。例如,应用程序的主线程希望在负责启动框架服务的线程已经启动所有的框架服务之后执行。

 

CountDownlatch 原理

通过一个计数器来实现的,计数器的初始值为线程的数量。每当一个线程完成了自己的任务后,计数器的值就相应的减1 。当计数器到达 0 时,表示所有的线程都已完成了任务,然后在闭锁上等待的线程就可以恢复执行任务。

 

 

CountDownLatch的用法

CountDownlacth典型用法1:

某一个线程开始运行前等待n个线程执行完毕。将 CountDownLatch 的计数器初始化为n (new CountDownLatch(n)),每当一个任务线程执行完毕,就将计数器减1  (countdownlatch.countDown() ),当计数器的值变为0时,在CountDownLatch上 await() 的线程就会被唤醒。一个典型应用场景就是启动一个服务时,主线程需要等待多个组件加载完毕,之后再继续执行。

需要注意的是:

await有多种方法,无限制时间等待    和 有限制时间等待

public void await() throws InterruptedException {
     sync.acquireSharedInterruptibly(1);
}

public boolean await(long timeout, TimeUnit unit)
     throws InterruptedException {
     return sync.tryAcquireSharedNanos(1, unit.toNanos(timeout));
}

 

使用例子:

    public static void main(String[] args) throws InterruptedException {
        CountDownLatch latch = new CountDownLatch(10);

        for (int i = 0; i < 10; i++) {
            new Thread(new Runnable(){
                @Override
                public void run(){
                    System.out.println(Thread.currentThread().getName()+"  正在运行");
                    try {
                        Thread.sleep(3000);
                    }catch( Exception e){
                        e.printStackTrace();
                    } finally {
                        latch.countDown();
                    }
                }
            }).start();
        }
        System.out.println("等待子线程运行结束");
        latch.await();
        //latch.await(10, TimeUnit.SECONDS);
        System.out.println("子线程运行结束");
    }

 

 

CountDownLatch典型用法2:

实现多个线程开始执行任务的最大并行性。注意是并行性,不是并发,强调的是多个线程在某一时刻同时开始执行。类似于赛跑,将多个线程放到起点,等待发令枪响,然后同时开跑。做法是初始化一个共享的CountDownLatch(1),将其计数器初始化为1,多个线程在开始执行任务前首先 coundownlatch.await(),当主线程调用 countDown() 时,计数器变为0,多个线程同时被唤醒。

 子线程等待主线程处理完毕开始处理,子线程处理完毕后,主线程输出

class MyRunnable implements Runnable {
 
    private CountDownLatch countDownLatch;
 
    private CountDownLatch await;
 
    public MyRunnable(CountDownLatch countDownLatch, CountDownLatch await) {
        this.countDownLatch = countDownLatch;
        this.await = await;
    }
 
    @Override
    public void run() {
        try {
            countDownLatch.await();// 所有线程在此进入等待状态
            System.out.println("子线程" +Thread.currentThread().getName()+ "处理自己事情");
            Thread.sleep(1000);
            await.countDown();// 所有线程在此进入等待状态
       } catch (InterruptedException e) { e.printStackTrace(); } } }
public static void main(String[] args) throws InterruptedException {
        CountDownLatch countDownLatch = new CountDownLatch(1);
        CountDownLatch await = new CountDownLatch(5);
 
        for (int i=0; i< 5; i++) {
            new Thread(new MyRunnable(countDownLatch, await)).start();
        }
 
        System.out.println("主线程处理自己事情");
        Thread.sleep(3000);
        countDownLatch.countDown();//  主线程调用countDown() 时,计数器变为0,所有线程去做要做的事
        System.out.println("主线程处理结束");
        await.await();// 唤醒所有线程
        System.out.println("子线程处理完毕啦");
    }

 

在实时系统中的使用场景
实现最大的并行性:有时我们想同时启动多个线程,实现最大程度的并行性。例如,我们想测试一个单例类。如果我们创建一个初始计数器为1的CountDownLatch,并让其他所有线程都在这个锁上等待,只需要调用一次countDown()方法就可以让其他所有等待的线程同时恢复执行。
开始执行前等待N个线程完成各自任务:例如应用程序启动类要确保在处理用户请求前,所有N个外部系统都已经启动和运行了。
死锁检测:一个非常方便的使用场景是你用N个线程去访问共享资源,在每个测试阶段线程数量不同,并尝试产生死锁。

 

————————————————
版权声明:本文为CSDN博主「春风十里不及你」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq812908087/article/details/81112188

发表于
2020-01-07 21:08 
五色风车 
阅读(1254
评论(0
编辑 
收藏 
举报

 

版权声明:本文为yrjns原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/yrjns/p/12163745.html

并发之CountDownLatch用法详解 - 五色风车的更多相关文章

  1. caffe net 可视化工具,,层特征可视化 – Sanny.Liu-CV&&ML

    caffe net 可视化工具,,层特征可视化 caffe Net 可视化工具,层特征可视化 1.只用网络在线 […]...

  2. sklearn.svm包中的SVC(kernel=”linear“)和LinearSVC的区别 – ZealouSnesS

    sklearn.svm包中的SVC(kernel=”linear“)和LinearSVC的区别 2017-10 […]...

  3. Oracle触发器之系统触发器 – 不酷也要写代码

    Oracle触发器之系统触发器 系统触发器 可以用系统触发器记录一些ddl的数据操作或者是数据库的登录 或者登 […]...

  4. eclipse中常用快捷键 – 龙吟浅笑

    eclipse中常用快捷键 在网上找了一些eclipse的常用快捷键,粘贴于此,以供参考 Ctrl+D: 删除 […]...

  5. 基于puppeteer模拟登录抓取页面 – JerremyZhang

    基于puppeteer模拟登录抓取页面 2018-05-08 19:49  JerremyZhang  阅读( […]...

  6. Java 集合系列06之 Vector详细介绍(源码解析)和使用示例 – 如果天空不死

    Java 集合系列06之 Vector详细介绍(源码解析)和使用示例   概要 学完ArrayList和Lin […]...

  7. 微信公众号页面无法唤起输入框 – 苦逼的IT男

    微信公众号页面无法唤起输入框        在做的微信公众号小项目中微信公众号页面无法唤起输入框?   遇到这 […]...

  8. office365的开发者训练营,免费,在微软广州举办 – 叶伟民

    office365的开发者训练营,免费,在微软广州举办 2018-09-16 19:57  叶伟民  阅读(1 […]...

随机推荐

  1. [Cisco]一些思科路由器命令,不断更新……

    trace xxx.xxx.xxx.xxx show int f0/0 show cdp neighbors […]...

  2. 生活杂谈之万网域名注册过程详解

    什么叫域名?我相信有很多人不了解这个名词,”域名”在百度百科解释,是由一串用点分隔的名字组成的Internet […]...

  3. Jenkins 基础篇-凭证配置

    我们在正式开始使用 Jenkins 之前还要先配置一些凭证,这些凭证的作用主要是用于认证,例如我们要从代码仓库 […]...

  4. Fiddler使用教程(转)

    Fiddler是最强大最好用的Web调试工具之一,你对HTTP协议越了解, 你就能越掌握Fiddler的使用方 […]...

  5. spring boot报错:Invalid bound statement (not found): com.

    经检查发现mapper的namespace没写全导致的    正确应该写成这样就可以了:  ...

  6. 面试题2:BAT及各大互联网公司2014前端笔试面试题:HTML/CSS篇

    BAT及各大互联网公司2014前端笔试面试题:HTML/CSS篇 Html篇: 1.你做的页面在哪些流览器测试 […]...

  7. Tomcat——简介、目录结构等

    软件的体系结构 1.C/S结构 Client/Server 客户端/服务端 *VB,Delphi,VC++,C […]...

  8. idea下载安装教程 – 学java及框架的菜鸡

    idea下载安装教程 下载教程 官网下载地址:https://www.jetbrains.com/idea/d […]...

展开目录

目录导航