并发之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. 开发人员、程序员与计算机科学家三者之间的区别 – 狼人:-)

    开发人员、程序员与计算机科学家三者之间的区别 2011-09-23 23:00  狼人:-)  阅读(544) […]...

  2. Serverless + Egg.js 后台管理系统实战

    本文将介绍如何基于 Egg.js 和 Serverless 实现一个后台管理系统 作为一名前端开发者,在选择 […]...

  3. ssdb常用知识点 – 婵婵

    ssdb常用知识点 ssdb备份与恢复  http://ssdb.io/docs/zh_cn/backup.h […]...

  4. 人工智能领域技术,主要包含了哪些核心技术? – sea的博客

    人工智能领域技术,主要包含了哪些核心技术? 人工智能领域技术,主要包含了哪些核心技术? 从语音识别到智能家居, […]...

  5. pdf 图片提取 – lydstory

    pdf 图片提取 axPdfview2  .net...

  6. scrum经验 – 风言枫语

    scrum经验 Scrum是基于过程控制理论的经验方法,倡导自组织团队;其运行框架核心是迭代增量型并行开发,也 […]...

  7. 用 jQuery 实现表单验证(摘抄) – wangzhiliang

    用 jQuery 实现表单验证(摘抄) ——选自《锋利的jQuery》(第2版)第5章的例题  5.1.5 表 […]...

  8. 系统架构设计师软考复盘 – 飘移

    系统架构设计师软考复盘 在大学时就听室友说他舅舅拿到了高级架构师的资质,是 IT 行业比较难拿的证书。当时只知 […]...

随机推荐

  1. 【雕爷学编程】Arduino动手做(44)—类比霍尔传感器

    雕爷学编程,Arduino动手做,开源硬件,创客传感器,类比霍尔传感器 37款传感器与模块的提法,在网络上广泛 […]...

  2. 服务发现组件之 — Eureka

    前言 现在流行的微服务体系结构正在改变我们构建应用程序的方式,从单一的单体服务转变为越来越小的可单独部署的服务 […]...

  3. Java 操作 Json

    1. 使用Gson构建Json 初始化 JsonObject jsonObject = new JsonObj […]...

  4. ios证书安装和打包流程

    iOS开发流程 1.拿到源文件     2文件目录大致名字      一、证书配置 参考网站:http://w […]...

  5. EdgeX Foundry试运行

    EdgeX Foundry试运行 简介 EdgeX Foundry是一个由Linux基金会发起的,且厂商中立的 […]...

  6. Linux应用程序打包

        原文地址:http://blog.solrex.cn/articles/packaging-1-src […]...

  7. python socket编程制作后门木马(原创)

    懒得解释怎么写出来的,直接上代码   Client:  简单连接服务器 #!/usr/bin/python # […]...

  8. 面试系列——Mysql索引

    面试系列——Mysql索引 1、索引分类 Hash索引Hash 索引查询效率很高,时间复杂度O(1)。Mysq […]...

展开目录

目录导航