闭锁

CountDownLatch概念

CountDownLatch是一个同步工具类,用来协调多个线程之间的同步,或者说起到线程之间的通信(而不是用作互斥的作用)。

CountDownLatch能够使一个线程在等待另外一些线程完成各自工作之后,再继续执行。使用一个计数器进行实现。计数器初始值为线程的数量。当每一个线程完成自己任务后,计数器的值就会减一。当计数器的值为0时,表示所有的线程都已经完成一些任务,然后在CountDownLatch上等待的线程就可以恢复执行接下来的任务。

CountDownLatch的用法

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

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

CountDownLatch的不足

CountDownLatch是一次性的,计算器的值只能在构造方法中初始化一次,之后没有任何机制再次对其设置值,当CountDownLatch使用完毕后,它不能再次被使用。

CountDownLatch(倒计时计算器)使用说明

方法说明

public void countDown()

  递减锁存器的计数,如果计数到达零,则释放所有等待的线程。如果当前计数大于零,则将计数减少.

public boolean await(long timeout,TimeUnit unit) throws InterruptedException

  使当前线程在锁存器倒计数至零之前一直等待,除非线程被中断或超出了指定的等待时间。如果当前计数为零,则此方法立刻返回true值。

  如果当前计数大于零,则出于线程调度目的,将禁用当前线程,且在发生以下三种情况之一前,该线程将一直出于休眠状态:

  由于调用countDown()方法,计数到达零;或者其他某个线程中断当前线程;或者已超出指定的等待时间。

  • 如果计数到达零,则该方法返回true值。
  • 如果当前线程,在进入此方法时已经设置了该线程的中断状态;或者在等待时被中断,则抛出InterruptedException,并且清除当前线程的已中断状态。
  • 如果超出了指定的等待时间,则返回值为false。如果该时间小于等于零,则该方法根本不会等待。

参数:

  timeout-要等待的最长时间

  unit-timeout 参数的时间单位

返回:

  如果计数到达零,则返回true;如果在计数到达零之前超过了等待时间,则返回false

抛出:

  InterruptedException-如果当前线程在等待时被中断

例子1:

  主线程等待子线程执行完成在执行

java.util.concurrent包中提供了多种并发容器类来改进同步容器的性能。ContDownLatch是一个同步辅助类,在完成某些运算时,只有其他所有线程的运算全部完成,当前运算才继续执行,这就叫闭锁。看下面代码

这段代码就是10个线程同时去输出5000以内的偶数,然后在主线程那里计算执行时间。其实这是计算不了那10个线程的执行时间的,因为主线程与这10个线程也是同时执行的,可能那10个线程才执行到一半,主线程就已经输出“耗费时间为x秒”这句话了。所有要想计算这10个线程执行的时间,就得让主线程先等待,等10个分线程都执行完了才能执行主线程。这就要用到闭锁。看如何使用:

使用CountDownLatch

public class TestCountDownLatch2 {public static void main(String[] args) {CountDownLatch latch = new CountDownLatch(10);LatchDemo2 ld = new LatchDemo2(latch);Long start = System.currentTimeMillis();for (int i = 0; i < 10; i++) {new Thread(ld).start();}try {latch.await();//这10个线程执行完之前先等待} catch (InterruptedException e) {}long end = System.currentTimeMillis();System.out.println("耗费时间为:" + (end - start));}
}
class LatchDemo2 implements Runnable {private CountDownLatch countDownLatch;public LatchDemo2(CountDownLatch countDownLatch) {this.countDownLatch = countDownLatch;}@Overridepublic void run() {synchronized (this) {try {for (int i = 0; i < 5000; i++) {if (i % 2 == 0) {System.out.println(i);}}} finally {//每执行完一个就递减一个countDownLatch.countDown();}}}
}

如上代码,主要就是用latch.countDown()latch.await()实现闭锁

CountDownLatch的理解和使用相关推荐

  1. CountDownLatch的理解和使用 多线程同步器

    CountDownLatch的理解和使用 在笔者想要了解Thrift时候,找到一个博主写的系统间通信技术的架构设计,在了解和学习的过程中遇到很多小问题和基础知识,自己还是不够清楚,就查询和总结下. 因 ...

  2. Java笔记-对CountDownLatch的理解(对比Qt中的QSemaphore)含实例

    首先在CountDownLatch,这个东西基本上和信号量是一样的,这个CountDownLatch要设置一个初值,这个值一般是个正值,可以对这个CountDownLatch进行countDown() ...

  3. 理解Java并发编程:CountDownLatch解析

    文章目录 快速入门 实战案例 案例1 案例2 小结 在一些场景下,我们可能有多个任务线程,其中某个任务依赖其他任务执行完成才能执行.如何实现? 有人可能说可以使用Thread类的join方法,join ...

  4. CountDownLatch与CyclicBarrier使用与理解

    CountDownLatch 我之前遇到过这么一个需求:"客户端同时下载视频.音频和大量试题压缩包".我让线程池分配三个线程同时开启下载三类数据,等到它们都下载完成时再进行数据整合 ...

  5. Java并发编程系列之CountDownLatch用法及详解

    背景 前几天一个同事问我,对这个CountDownLatch有没有了解想问一些问题,当时我一脸懵逼,不知道如何回答.今天赶紧抽空好好补补.不得不说Doug Lea大师真的很牛,设计出如此好的类. 1. ...

  6. 【JDK】JDK源码分析-CountDownLatch

    概述 CountDownLatch 是并发包中的一个工具类,它的典型应用场景为:一个线程等待几个线程执行,待这几个线程结束后,该线程再继续执行. 简单起见,可以把它理解为一个倒数的计数器:初始值为线程 ...

  7. 如何判断线程池已经执行完所有任务了?

    作者 | 磊哥 来源 | Java面试真题解析(ID:aimianshi666) 转载请联系授权(微信ID:GG_Stone) 很多场景下,我们需要等待线程池的所有任务都执行完,然后再进行下一步操作. ...

  8. 京东暑期实习面经(已OC)

    京东实习一面面经(2021年3月2日下午4点) 体验 我能说-面试官的周围声音太大了吗?hhh面试过程中我也不好提,旁边人非常多,噪音很大,多少受到些干扰. 除了这一点的话,其他体验都非常好.特别让我 ...

  9. Java_JUC_CountDownLatch/CyclicBarrier循环屏障/Semaphore信号灯/枚举

    Java_JUC_CountDownLatch/CyclicBarrier循环屏障/Semaphore信号灯/枚举 java常用的juc api (一)枚举 1.1 代码实现 package juca ...

最新文章

  1. 字节跳动教育裁员:赔付方式N+2
  2. Linux命令之du
  3. HDU 4763 Theme Section(KMP+枚举公共前后缀)
  4. Myeclipse创建Maven项目提示:Creating maven-archetype-webapp has encountered a problem. 解决方法
  5. 一朝读码深似海,不读源码薪难升!读懂.NET5源码,到底多重要?
  6. 关于H264通过RTP传输的打包方式
  7. 运算符--位移运算符和一些其他运算符
  8. 进程通信方法的特点以及使用场景
  9. 红皮书:变量、作用域和内存问题(四)
  10. tar 整个linux系统,linux下tar解压
  11. android卡片 弹簧滑动,一种通用式弹簧卡扣的制作方法
  12. Excel添加下拉选项菜单的方法
  13. idea打开文件显示多行,不隐藏
  14. LeetCode874 模拟行走机器人
  15. geoserver配置SQL图层 cql_filter模糊查询
  16. Cannot use import statement outside a module
  17. miniGui交叉编译
  18. 欢迎中文社区新版主@黄念刚
  19. canvas画笔自定义笔触
  20. python退出交互_python交互界面的退出方法

热门文章

  1. 基础入门_Python-内建函数.运维开发中eval内建函数的最佳实践?
  2. Hash(LCP) || 后缀数组 LA 4513 Stammering Aliens
  3. JavaScript复制数组
  4. iphone3G恢复到3.1.2遇到的问题
  5. 探讨程序员如何学习你不熟悉的技术,以及用什么样的方法去学习.
  6. halcon自动对焦算法
  7. ABB机器人套接口通信 机器人部分
  8. springboot数据源不正确_Spring MVC 到 Spring Boot 的简化之路
  9. html5清除手机页面缓存文件夹,WebView自动缓存-清除缓存
  10. Wampserver之 virtualHost