当前位置: 首页 > news >正文

并发工具类-CountDownLatch

CountDownLatch 是 Java 中提供的一种非常有用的并发工具类,位于 java.util.concurrent 包中。它可以使一个或多个线程等待其他线程完成一组特定的操作后再继续执行。CountDownLatch 通过维护一个计数器来实现这一点,计数器的初始值由构造函数设定。每当一个线程完成了它的工作后,就调用 countDown() 方法将计数器减 1,当计数器减为 0 时,所有等待的线程都会被唤醒并继续执行。

1. CountDownLatch 的基本概念

1.1 核心思想

CountDownLatch 的核心思想是线程之间的协调与同步,它允许一个或多个线程等待一组事件发生后再执行某个操作。主要通过以下机制实现:

  • 一个或多个线程调用 await() 方法进入等待状态,直到计数器的值变为 0。
  • 其他线程通过 countDown() 方法递减计数器的值,每次递减操作代表一个任务完成。
  • 当计数器的值变为 0 时,所有在 await() 方法上等待的线程会被唤醒并继续执行。
1.2 工作流程
  • 初始阶段:计数器值(count)是通过构造方法设置的,表示有多少个任务需要完成。
  • 任务完成:每个线程执行完一个任务后调用 countDown() 方法,使计数器减 1。
  • 等待线程:调用 await() 的线程会阻塞,直到计数器的值变为 0,表示所有任务都已完成。

2. CountDownLatch 的构造方法和主要方法

2.1 构造方法

CountDownLatch 只有一个构造方法:

public CountDownLatch(int count)
  • count:传入的参数表示计数器的初始值,代表有多少个任务需要完成。
2.2 主要方法
  1. await()
       - 作用:调用该方法的线程会进入等待状态,直到计数器的值变为 0,或者线程被中断。
       - 示例:
         java      latch.await();      

  2. countDown()
       - 作用:将计数器的值减 1。每次调用表示一个任务完成,当计数器的值变为 0 时,所有调用了 await() 的线程会被唤醒。
       - 示例:
         java      latch.countDown();      

  3. await(long timeout, TimeUnit unit)
       - 作用:调用该方法的线程会等待指定的时间。如果计数器在指定时间内变为 0,线程继续执行;否则,线程会超时返回。
       - 示例:
         java      latch.await(5, TimeUnit.SECONDS);      

  4. getCount()
       - 作用:返回当前计数器的值。
       - 示例:
         java      long count = latch.getCount();      

3. CountDownLatch 的使用场景

CountDownLatch 适合用于多线程协调场景,以下是一些典型的使用场景:

3.1 主线程等待多个子线程执行完毕

当主线程需要等待多个子线程执行完成后再继续执行时,可以使用 CountDownLatch 来实现。例如,在主线程中创建多个子线程去处理不同的任务,主线程调用 await() 方法等待所有子线程完成后再继续执行后续操作。

3.2 实现并行任务后合并结果

在分布式系统或高并发场景中,常常需要将一个任务拆分为多个子任务,并发执行后再合并结果。CountDownLatch 可以保证主线程等待所有并发子任务完成后,再对结果进行汇总。

3.3 模拟并发压力测试

在某些并发压力测试中,可能需要多个线程同时开始执行任务。CountDownLatch 可以用来控制多个线程同时开始某个操作,从而模拟高并发场景。

4. CountDownLatch 的实际使用示例

以下是一个使用 CountDownLatch 的典型示例,模拟主线程等待多个子线程完成任务。

import java.util.concurrent.CountDownLatch;public class CountDownLatchExample {public static void main(String[] args) throws InterruptedException {// 创建 CountDownLatch,计数器初始值为3,表示有3个任务需要完成CountDownLatch latch = new CountDownLatch(3);// 启动3个子线程for (int i = 1; i <= 3; i++) {new Thread(new Worker(latch, "Worker-" + i)).start();}// 主线程等待,直到所有子线程执行完任务latch.await();System.out.println("All workers have finished. Main thread proceeding.");}
}class Worker implements Runnable {private CountDownLatch latch;private String name;public Worker(CountDownLatch latch, String name) {this.latch = latch;this.name = name;}@Overridepublic void run() {try {// 模拟工作过程System.out.println(name + " is working.");Thread.sleep((long) (Math.random() * 1000)); // 模拟工作时间System.out.println(name + " finished work.");} catch (InterruptedException e) {Thread.currentThread().interrupt();} finally {// 任务完成,调用 countDown() 方法,使计数器减1latch.countDown();}}
}

解释

  • 我们创建了一个 CountDownLatch 对象,计数器初始值为 3。
  • 创建了 3 个子线程(模拟 3 个工作者),每个线程执行完任务后调用 countDown(),将计数器减 1。
  • 主线程调用 await() 方法,等待所有子线程完成任务。
  • 当计数器变为 0 时,主线程继续执行。

5. CountDownLatch 的应用场景举例

5.1 多服务启动协调

在微服务架构或分布式系统中,一个服务的启动可能依赖于其他服务。如果某个服务需要等待多个依赖服务启动完成后才能启动,可以使用 CountDownLatch 来协调启动流程。

public class ServiceStartupExample {private static final int NUM_OF_SERVICES = 3;private static CountDownLatch latch = new CountDownLatch(NUM_OF_SERVICES);public static void main(String[] args) throws InterruptedException {// 启动多个依赖服务new Thread(new Service("Service-1", 2000)).start();new Thread(new Service("Service-2", 3000)).start();new Thread(new Service("Service-3", 4000)).start();// 等待所有服务启动完毕latch.await();System.out.println("All dependent services are up. Starting main service.");}static class Service implements Runnable {private String serviceName;private int startupTime;public Service(String serviceName, int startupTime) {this.serviceName = serviceName;this.startupTime = startupTime;}@Overridepublic void run() {try {// 模拟服务启动时间System.out.println(serviceName + " is starting...");Thread.sleep(startupTime);System.out.println(serviceName + " is up.");} catch (InterruptedException e) {Thread.currentThread().interrupt();} finally {// 启动完成,计数器减1latch.countDown();}}}
}

在该示例中,主线程等待所有依赖服务启动完成后才继续启动主服务。CountDownLatch 用于协调各个依赖服务的启动过程,确保它们按预期的顺序和时间依赖性执行。

6. CountDownLatch 的线程安全与性能分析

6.1 线程安全

CountDownLatch 的内部实现是线程安全的,它通过内部的锁机制和原子操作来确保计数器的递减操作是安全的。每次调用 countDown() 时,计数器会以原子的方式递减,避免了竞态条件的出现。此外,await() 方法会保证线程阻塞直到计数器变为 0,从而确保多个线程可以同步进行操作。

6.2 性能
  • 开销CountDownLatch 的开销主要在于线程的阻塞与唤醒。对于少量线程的协调,这些开销通常是可以忽略的。然而,在高并发场景下,如果涉及大量线程调用 await() 进行阻塞等待,可能会导致一些性能开销。
  • 使用场景:由于 CountDownLatch 只能使用一次,计数器不能被重置,因此适用于一次性协调多个线程的场景。如果需要多个阶段的同步或复用,可以使用

CyclicBarrier 或其他并发工具。

7. CountDownLatch 与其他并发工具类的比较

  • CyclicBarrierCyclicBarrierCountDownLatch 都用于协调多个线程的执行,但 CyclicBarrier 允许计数器重置,适用于循环使用的场景,而 CountDownLatch 是一次性使用的。
  • SemaphoreSemaphore 用于控制对资源的并发访问,限制同时访问资源的线程数。而 CountDownLatch 则侧重于等待其他线程完成任务。

8. 总结

CountDownLatch 是一个非常有用的并发工具类,能够帮助多个线程之间协调执行顺序。它的工作原理基于计数器,线程可以通过 await() 方法等待,直到其他线程通过 countDown() 完成任务。CountDownLatch 在实际编程中的应用场景非常广泛,包括并发任务的协调、多线程测试中的并发压力模拟以及服务依赖启动等。

主要优点:

  • 易于理解和使用,适用于需要一次性等待多个线程完成任务的场景。
  • 线程安全,内部使用锁机制保证计数器的递减和线程阻塞的安全性。

缺点:

  • 计数器不可重置,因此只能使用一次,如果需要重用,应该考虑使用 CyclicBarrier 等工具类。

相关文章:

并发工具类-CountDownLatch

CountDownLatch 是 Java 中提供的一种非常有用的并发工具类&#xff0c;位于 java.util.concurrent 包中。它可以使一个或多个线程等待其他线程完成一组特定的操作后再继续执行。CountDownLatch 通过维护一个计数器来实现这一点&#xff0c;计数器的初始值由构造函数设定。每当…...

进程的重要函数

进程的重要函数: fork函数 了解fork函数 通过调用fork()函数&#xff0c;则会产生一个新的进程。调用fork()函数的进程叫做 父进程&#xff0c;产生的新进程则为子进程。 其编码过程: 1.函数功能: 函数头文件 #include <sys/types.h> #include <unistd.h> 函数…...

python 实现average median平均中位数算法

average median平均中位数算法介绍 平均&#xff08;Mean&#xff09;和中位数&#xff08;Median&#xff09;是统计学中常用的两个概念&#xff0c;用于描述一组数据的中心趋势&#xff0c;但它们并不是算法&#xff0c;而是数据处理的结果。不过&#xff0c;我可以解释如何…...

HTML概述

1. HTML概述 1.1 HTML定义 HTML超文本标记语言&#xff0c;其中超文本是链接&#xff0c;标记也叫标签&#xff08;即带尖括号的文本&#xff09;。 1.2 HTML基本骨架 HTML基本骨架是网页模板。 <html><head><title>网页的标题</title></head&…...

【FFT】信号处理——快速傅里叶变换【通俗易懂】

快速傅里叶变换&#xff08;Fast Fourier Transform, FFT&#xff09;是一种用于将信号从时间域转换到频率域的算法。 傅里叶变换的核心思想是&#xff1a;任何周期性信号都可以分解成多个不同频率的正弦波或余弦波的叠加。 简单来说&#xff0c;FFT可以帮助我们理解一个信号…...

电脑升级WIN11之后需要注意哪些东西

1.记事本&#xff0c;在前单位时&#xff0c;电脑升级后&#xff0c;记事本会需要手动更新&#xff0c;或手动安装 2.任务栏&#xff0c;WIN11默认任务栏在中间位置&#xff0c;想要调成WIN10一样的位置&#xff0c;分享两个方法 拖拽法&#xff08;适用于Windows 11 2022年1…...

GEE 教程:利用sentinel-5p数据进行长时序CO一氧化碳的监测分析并结合夜间灯光数据分析

目录 简介 数据 哨兵5号 NOAA/VIIRS/DNB/MONTHLY_V1/VCMCF 函数 ui.Chart.image.series(imageCollection, region, reducer, scale, xProperty) Arguments: Returns: ui.Chart 代码 结果 简介 利用sentinel-5p数据进行长时序CO一氧化碳的监测分析并结合夜间灯光数据…...

【教程】鸿蒙ARKTS 打造数据驾驶舱---前序

鸿蒙ARKTS 打造数据驾驶舱 ​ 前面2章我介绍了如何通过定义View绘制箭头以及圆形进度&#xff0c;初步了解了鸿蒙如何进行自定义View。接下来我将通过我最近在带的一个VUE的项目&#xff0c;简单实现了几个鸿蒙原生页面。帮助大家快速上手纯血鸿蒙开发. 本项目基于Api11Stage模…...

Html css样式总结

1.Html css样式总结 CSS 定义 中文名称&#xff1a;层叠样式表 。 英文全称&#xff1a;Cascading Style Sheets &#xff0c;简称CSS。在网页制作时采用CSS技术&#xff0c;可以有效地对页面的布局、字体、颜色、背景和其它效果实现更加精确的控制。 &#xff08;1&#xff09…...

决策树基础概论

1. 概述 在机器学习领域&#xff0c;决策树&#xff08;Decision Tree&#xff09; 是一种高度直观且广泛应用的算法。它通过一系列简单的是/否问题&#xff0c;将复杂的决策过程分解为一棵树状结构&#xff0c;使得分类或回归问题的解决过程直观明了。决策树的最大特点在于可…...

Spring Boot集成Akka Cluster快速入门Demo

1.什么是Akka Cluster&#xff1f; Akka Cluster将多个JVM连接整合在一起&#xff0c;实现消息地址的透明化和统一化使用管理&#xff0c;集成一体化的消息驱动系统。最终目的是将一个大型程序分割成若干子程序&#xff0c;部署到很多JVM上去实现程序的分布式并行运算&#xf…...

django学习入门系列之第十点《A 案例: 员工管理系统10》

文章目录 12 管理员操作12.4 密码加密12.5 获取对象&#xff08;防止id错误--编辑界面等&#xff09;12.6 编辑管理员12.7 重置密码 往期回顾 12 管理员操作 12.4 密码加密 密码不应该以明文的方式直接存储到数据库&#xff0c;应该加密才放进去 定义一个md5的方法&#xff…...

Unity实战案例全解析:PVZ 植物卡片状态分析

Siki学院2023的PVZ免费了&#xff0c;学一下也坏 卡片状态 卡片可以有三种状态&#xff1a; 1.阳光足够&#xff0c;&#xff08;且cd好了可以种植&#xff09; 2.阳光不够&#xff0c;&#xff08;cd&#xff1f;好了&#xff1a;没好 &#xff08;三目运算符&#xff09;&…...

判断变量是否为有限数字(非无穷大或NaN)math.isfinite() 判断变量是否为无穷大(正无穷大或负无穷大)math.isinf()

【小白从小学Python、C、Java】 【考研初试复试毕业设计】 【Python基础AI数据分析】 判断变量是否为有限数字&#xff08;非无穷大或NaN&#xff09; math.isfinite() 判断变量是否为无穷大&#xff08;正无穷大或负无穷大&#xff09; math.isinf() 请问关于以下代码表述错误…...

idea使用阿里云服务器运行jar包

说明&#xff1a;因为我用的阿里云服务器不是自己的&#xff0c;所以一些具体的操作可能不太全面。看到一个很完整的教程&#xff0c;供参考。 0. 打包项目 这里使用的是maven打包。 在pom.xml中添加以下模块。 <build><plugins><plugin><groupId>org…...

解决nginx代理SSE接口的响应没有流式返回

目录 现象原来的nginx配置解决 现象 前后端分离的项目&#xff0c;前端访问被nginx反向代理的后端SSE接口&#xff0c;预期是流式返回&#xff0c;但经常是很久不响应&#xff0c;一响应全部结果一下子都返回了。查看后端项目的日志&#xff0c;响应其实是流式产生的。推测是n…...

11 - TCPClient实验

在上一个章节的UDP通信测试中&#xff0c;尽管通信的实现过程相对简洁&#xff0c;但出现了通信数据丢包的问题。因此&#xff0c;本章节将基于之前建立的WIFI网络连接&#xff0c;构建一个基础的TCPClient连接机制。我们利用网络调试助手工具来发送数据&#xff0c;测试网络通…...

React框架搭建,看这一篇就够了,看完你会感谢我

传统搭建框架的方式 在2024年以前&#xff0c;我们构建框架基本上采用官方脚手架&#xff0c;但是官方脚手架其实大概率都不符合我们的项目要求&#xff0c;搭建完了以后往往需要再继续集成一些第三方的包。这时候又会碰到一些版本冲突&#xff0c;配置教程等&#xff0c;往往…...

【rust】rust条件编译

在c语言中&#xff0c;条件编译是一个非常好用的功能&#xff0c;那么rust中如何实现条件编译呢? rust的条件编译需要两个部分&#xff0c;一个是fratures&#xff0c;另一个是cfg。Cargo feature是一个非常强大的功能&#xff0c;可以提供条件编译和可选依赖项的高级特性&…...

一键文本提示实现图像对象高质量剪切与透明背景生成

按照提示词裁剪 按照边框裁剪 要实现您描述的功能,即通过一个文本提示就能自动从图片中切割出指定的对象并生成一个带有透明背景的新图像,这需要一个结合了先进的计算机视觉技术和自然语言处理能力的系统。这样的系统可以理解输入的文本指令,并将其转化为对图像内容的精确分…...

T型翼/尾板导向的穿浪双体船姿态控制【附代码】

✨ 长期致力于穿浪双体船、T型翼、尾板、多自由度姿态控制、舒适性评估研究工作&#xff0c;擅长数据搜集与处理、建模仿真、程序编写、仿真设计。 ✅ 专业定制毕设、代码 ✅ 如需沟通交流&#xff0c;点击《获取方式》 &#xff08;1&#xff09;动态水翼升力模型与耦合运动方…...

基于XGBoost与SHAP的分子气味预测:从特征工程到可解释性分析

1. 项目概述与核心价值在香水设计、食品风味工业乃至环境监测领域&#xff0c;一个核心且持久的挑战是&#xff1a;如何从分子的化学结构出发&#xff0c;准确预测其气味&#xff1f;这不仅仅是化学家或调香师的直觉游戏&#xff0c;更是一个复杂的、高维度的模式识别问题。传统…...

DIY复刻经典:Texar Audio Prism动态处理器克隆套件全攻略

1. 项目概述&#xff1a;Texar Audio Prism 克隆套件如果你在专业音频圈子里混过一段时间&#xff0c;尤其是对上世纪八九十年代那些经典的、带点“魔法”色彩的外置动态处理器感兴趣&#xff0c;那么“Texar Audio Prism”这个名字你大概率不会陌生。它不是最常见的1176或者LA…...

微信小程序3D开发框架技术对比:XR-Frame与threejs-miniprogram

随着微信小程序逐步支持3D渲染与AR能力&#xff0c;开发者面临两个主要官方方案&#xff1a;自研的XR-Frame和适配Three.js的threejs-miniprogram。本文将从架构设计、渲染机制、功能集成、开发模式及适用场景等维度进行技术分析&#xff0c;为技术选型提供参考。一、XR-Frame&…...

毕业设计 yolov11骨折检测医疗辅助系统(源码+论文)

文章目录 0 前言1 项目运行效果2 课题背景2.1 研究背景2.2 国内外研究现状2.3 研究意义 3 设计框架&#xff08;骨折检测系统设计框架说明&#xff09;3.1. 系统架构图3.2. 技术选型3.2.1 核心组件3.2.2 辅助工具 3.3. 核心模块设计3.3.1 YOLO模型训练模块训练流程图关键伪代码…...

自制极低频电流探头:负电阻补偿原理与低频方波测量实践

1. 项目概述&#xff1a;为极低频电流测量而生在电子测试领域&#xff0c;电流探头是个再常见不过的工具&#xff0c;无论是排查开关电源的纹波&#xff0c;还是分析电机驱动的波形&#xff0c;都离不开它。但如果你尝试用市面上常见的电流探头去观察一个频率低至几赫兹&#x…...

YOLOv8晶圆体缺识别检测系统(项目源码+YOLO数据集+模型权重+UI界面+python+深度学习+环境配置)

摘要 晶圆制造过程中的缺陷检测是保证芯片良率的关键环节。本文基于YOLOv8目标检测算法&#xff0c;构建了一套针对晶圆表面9类典型缺陷的自动检测系统。所识别的缺陷类型包括&#xff1a;Center、Donut、Edge-Loc、Edge-Ring、Loc、Near-full、None、Random、Scratch。模型在…...

Gazebo Sim多旋翼控制:四轴飞行器动力学建模与PID调参

Gazebo Sim多旋翼控制&#xff1a;四轴飞行器动力学建模与PID调参 【免费下载链接】gz-sim Open source robotics simulator. The latest version of Gazebo. 项目地址: https://gitcode.com/gh_mirrors/gz/gz-sim Gazebo Sim是一款功能强大的开源机器人模拟器&#xff…...

BiliRoamingX:彻底解决B站体验限制的完整增强方案

BiliRoamingX&#xff1a;彻底解决B站体验限制的完整增强方案 【免费下载链接】BiliRoamingX-integrations BiliRoamingX integrations and patches powered by ReVanced. 项目地址: https://gitcode.com/gh_mirrors/bi/BiliRoamingX-integrations 你是否曾为B站的内容区…...

十年以上经验的建站公司推荐|策划强、落地稳的网站制作公司盘点

互联网时代&#xff0c;企业官网已从单纯的信息展示窗口升级为集品牌价值传递、用户体验连接与业务高效转化于一体的核心数字阵地。行业报告显示&#xff0c;优质官网可帮助企业线上转化率提升35%-60%&#xff0c;而低效官网则可能导致潜在客户大量流失。面对市场上众多的网站建…...