JUC下CountDownLatch详解

详细介绍

  CountDownLatch是Java并发包java.util.concurrent中提供的一个同步工具类,它允许一个或多个线程等待其他线程完成操作后再继续执行。这个工具类基于一个计数器,计数器的初始值可以由构造函数设定。线程调用countDown()方法会将计数器减1,而其他线程调用await()方法会阻塞,直到计数器为0。这在多线程协作中非常有用,特别是在需要等待某些条件达成(比如所有子任务完成)之后,再继续执行后续操作的场景。

核心API
  • 构造方法CountDownLatch(int count),创建一个CountDownLatch实例,并初始化计数器为给定的count值。
  • countDown():递减计数器的值,如果计数器到达0,则释放所有等待的线程。
  • await():使当前线程等待,直到计数器达到0。这是一个阻塞方法,可被中断。
  • getCount():获取当前计数器的值,反映还有多少个countDown()调用才能到达零。
工作原理

  CountDownLatch通过一个共享的计数器实现线程间的同步。初始化时,计数器被赋予一个正整数值,表示需要等待的事件数量。每当一个线程完成一个事件(调用countDown()方法),计数器的值就减1。其他线程调用await()方法会阻塞,直到计数器减到0,此时所有阻塞的线程会被唤醒并继续执行。

实现细节

  CountDownLatch内部使用了AQS(AbstractQueuedSynchronizer)框架,这是Java并发包中的一个基础框架,用于构建锁和其他同步器。AQS维护了一个双向链表来管理等待线程,以及一个volatile变量表示同步状态(在CountDownLatch中即为计数器)。

适用场景拓展

除了上述基本使用场景,CountDownLatch还可以用于:

  • 压力测试:在性能测试或压力测试中,可以用来同步所有并发请求的开始时间,确保所有请求同时发起,以便准确测量系统在高并发下的表现。
  • 任务调度:在任务调度系统中,可以用来控制任务的开始时机,比如确保所有准备工作完成后再开始执行主要任务。
  • 系统关闭序列:在分布式系统中,可以用来控制优雅关闭流程,确保所有服务组件都完成特定的关闭操作后再完全关闭系统。
与CyclicBarrier的区别

虽然CountDownLatchCyclicBarrier都可以用于线程同步,但两者有本质区别:

  • 计数器的可重用性CountDownLatch的计数器只能递减到0,之后无法重置,是一次性使用的同步工具;而CyclicBarrier的屏障可以重置,适合多次重复的同步场景。
  • 同步点CountDownLatch是“一到多”的等待模型,一个或多个线程等待其他N个线程完成某项操作;而CyclicBarrier是“多对多”的等待模型,所有参与线程都等待彼此到达同一个同步点。

使用场景

  1. 1. 并行任务的同步

    在处理多个并行任务时,经常需要等待所有任务完成后再进行下一步操作,例如数据处理、资源初始化或结果汇总。CountDownLatch非常适合这类场景,通过它可以轻松实现任务的同步等待。

    示例:一个大数据处理应用需要将海量数据分割成多个小块,分配给多个线程并行处理,最后汇总各线程的处理结果。每个线程在完成自己的处理任务后调用countDown(),主线程则通过await()等待所有线程完成,之后执行结果汇总。

    2. 应用程序启动时的初始化同步

    在大型应用系统启动时,可能需要完成多个模块的初始化工作,这些初始化工作可以并行进行,但整个应用只有在所有初始化工作都完成之后才能进入就绪状态。

    示例:一个Web应用服务器启动时,需要初始化数据库连接池、加载配置文件、启动日志系统等多个步骤。通过为每个初始化任务分配一个CountDownLatch计数器,主线程可以等待所有初始化任务完成后再启动服务监听。

    3. 性能测试的同步启动

    在进行系统性能测试时,为了模拟真实的高并发场景,需要确保所有模拟客户端请求同时发起。CountDownLatch可以用来协调所有客户端线程,在计数器归零的一刻同时开始发送请求。

    示例:进行网站压力测试时,使用多个线程模拟用户访问,通过CountDownLatch确保所有线程在准备阶段完成后同时开始发送HTTP请求,以准确评估系统在高并发环境下的性能表现。

    4. 测试代码中的同步控制

    在单元测试或集成测试中,有时需要控制测试代码的执行顺序,确保某些代码段在其他线程完成特定操作后执行。CountDownLatch可以作为一种灵活的同步机制,帮助精确控制测试流程。

    示例:测试一个多线程交互的模块,需要确保一个线程修改数据后,另一个线程在检查数据之前,数据已完全准备好。利用CountDownLatch可以让测试线程在适当的时候开始执行验证逻辑。

    5. 分布式系统中的协调

    在分布式系统中,有时需要等待多个节点完成特定操作后,再进行下一步的协同工作。虽然CountDownLatch主要用于单JVM内线程同步,但在某些场景下,可以通过网络通信机制间接应用于分布式协调。

    示例:一个分布式任务调度系统,主节点分配任务给多个子节点执行,主节点需要等待所有子节点报告任务完成。虽然直接使用CountDownLatch跨节点不太现实,但可以设计类似机制,通过心跳检测或消息队列来模拟计数器的减少和等待逻辑。

使用示例:

假设有一个需求,需要启动多个线程执行不同的任务,但主程序需要等待所有这些任务完成后再继续执行后续逻辑。下面是一个使用CountDownLatch来实现这一需求的示例代码。

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class CountDownLatchExample {public static void main(String[] args) throws InterruptedException {// 创建一个固定大小的线程池ExecutorService executorService = Executors.newFixedThreadPool(3);// 初始化CountDownLatch,设置计数器为3,表示需要等待3个任务完成CountDownLatch latch = new CountDownLatch(3);System.out.println("Starting threads...");// 启动三个线程,每个线程执行完后调用countDown()方法for (int i = 0; i < 3; i++) {executorService.submit(() -> {try {Thread.sleep((long) (Math.random() * 1000)); // 模拟任务执行时间System.out.println("Task " + Thread.currentThread().getName() + " finished.");} catch (InterruptedException e) {Thread.currentThread().interrupt();} finally {// 任务完成,计数器减1latch.countDown();}});}// 主线程调用await(),等待所有任务完成latch.await();System.out.println("All tasks completed. Continuing with main program...");// 关闭线程池executorService.shutdown();}
}

解释说明

  1. 初始化CountDownLatch:首先创建一个CountDownLatch实例,并设置初始计数器值为3,意味着我们需要等待3个任务完成。

  2. 启动线程:通过线程池ExecutorService启动3个线程,每个线程执行一个简单的任务,模拟不同的处理时间。

  3. 计数器减1:每个线程在完成任务后调用latch.countDown(),这会将计数器减1,表明一个任务已经完成。

  4. 主线程等待:主线程调用latch.await(),此时主线程会阻塞,直到计数器减至0。这意味着所有任务都已完成。

  5. 继续执行:当所有任务完成,await()方法返回,主线程继续执行,打印出“所有任务完成”。

  6. 线程池关闭:最后,记得关闭线程池,释放资源。

通过这个示例,可以看出CountDownLatch在多线程协作中的重要作用,它提供了一种简单而有效的机制来同步多个线程的执行,确保所有任务完成后再进行下一步操作。

示例2:

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class CountDownLatchDemo {public static void main(String[] args) throws InterruptedException {int threadCount = 5;ExecutorService executorService = Executors.newFixedThreadPool(threadCount);CountDownLatch countDownLatch = new CountDownLatch(threadCount);for (int i = 0; i < threadCount; i++) {Runnable worker = new WorkerThread(countDownLatch, "Worker-" + i);executorService.execute(worker);}// 主线程调用await,等待所有worker线程完成countDownLatch.await();System.out.println("All workers completed their tasks.");executorService.shutdown();}
}class WorkerThread implements Runnable {private final CountDownLatch latch;private final String name;public WorkerThread(CountDownLatch latch, String name) {this.latch = latch;this.name = name;}@Overridepublic void run() {try {doWork();} finally {// 工作完成,计数器减1latch.countDown();}}private void doWork() {System.out.println(name + " is working...");try {Thread.sleep(1000);} catch (InterruptedException e) {Thread.currentThread().interrupt();}}
}

注意事项

1. 计数器不可重置

一旦创建了CountDownLatch实例并设置了初始计数值,这个计数器是不可逆的。也就是说,一旦计数器减到0,它将保持在0,不能再被重置为初始值。这意味着CountDownLatch主要用于一次性的同步事件,不适用于需要多次重置计数器的场景。对于需要循环使用的同步工具,可以考虑使用CyclicBarrier

2. 避免死锁

尽管CountDownLatch的设计旨在简化同步,但错误的使用仍然可能导致死锁。确保所有等待线程最终都能得到释放,避免在等待线程中调用会阻止其他线程调用countDown()的方法,否则可能会导致等待线程永远阻塞。

3. 线程中断处理

调用await()方法的线程可以被中断,这将导致InterruptedException被抛出。在处理中断时,应当妥善处理这个异常,比如记录日志、清理资源并优雅地结束线程。不要简单地吞掉这个异常,因为中断通常是用来控制线程生命周期的重要手段。

4. 资源管理

确保在不再需要时正确关闭或释放与CountDownLatch相关的资源,特别是当你在使用线程池或其他资源时。如果CountDownLatch是在一个大的应用上下文中使用,忘记释放资源可能会导致内存泄漏或其他资源占用问题。

5. 并发安全

虽然CountDownLatch自身是线程安全的,但使用它时仍需注意外部状态的并发访问。如果你在countDown()前后访问共享资源,务必确保这些访问是线程安全的,可能需要额外的同步措施。

6. 计数器初始化

在初始化CountDownLatch时,要确保计数器的初始值准确无误。错误的计数可能导致等待线程过早或过晚解除阻塞,从而破坏程序逻辑。

7. 性能考量

频繁的await()调用可能导致性能开销,特别是在计数器还未达到0时。如果等待的线程数量非常大,或者等待时间很长,可能需要考虑其他并发模型或优化等待逻辑。

8. 测试

在使用CountDownLatch的复杂并发程序中,测试变得尤为重要。使用单元测试和集成测试确保并发逻辑正确无误,特别关注边界条件和异常情况。

9. 文档和注释

清晰的文档和代码注释对于维护和理解使用了CountDownLatch的代码至关重要。说明每个CountDownLatch实例的作用、初始计数值以及为什么需要这样的同步机制,可以大大帮助未来的维护者。

优缺点

优点
  1. 简单易用CountDownLatch提供了一种直观且简洁的方式来同步线程,使得多个线程可以等待一个或多个事件的发生。它的API简单明了,易于理解和实现。

  2. 灵活性:它允许指定一个初始计数值,这意味着可以用来同步任意数量的事件或任务完成。这种灵活性使得CountDownLatch在多种并发场景下都能发挥作用。

  3. 高效同步:由于其基于低级别的同步原语(如AQS)实现,CountDownLatch提供了高效的线程同步机制,减少了不必要的线程上下文切换和等待时间。

  4. 集成方便:作为Java标准库的一部分,CountDownLatch与Java并发包的其他工具(如线程池ExecutorService)无缝集成,便于构建复杂的并发程序。

  5. 中断支持:调用await()的线程可以被中断,提供了处理长时间等待或取消操作的机制,增强了程序的响应性和可控性。

缺点
  1. 不可重置性:一旦计数器减至0,CountDownLatch就不能重置回初始值,这限制了它在需要重复同步事件的应用场景中的使用。相比之下,CyclicBarrier提供了一个可重置的计数器,更适合循环同步的需求。

  2. 潜在的死锁风险:虽然CountDownLatch本身不易导致死锁,但在复杂的并发环境中,如果使用不当,比如在countDown()执行路径上出现阻塞,可能导致等待线程永远无法被唤醒,形成事实上的死锁。

  3. 资源消耗:在某些情况下,特别是计数器初始值较大且等待线程数量多时,大量的线程等待可能会消耗较多的系统资源,包括内存和CPU时间(尤其是在上下文切换上)。

  4. 调试和维护难度:由于CountDownLatch引入了额外的线程同步逻辑,它可能增加程序的复杂性,特别是当涉及多个CountDownLatch实例交织使用时,调试和维护变得更加困难。

  5. 信息不透明CountDownLatch本身不提供关于哪些线程正在等待、哪些已经完成的直接信息,这在调试和监控并发程序时可能是个不足。

可能遇到的问题及解决方案

1. 死锁问题

问题描述:在使用CountDownLatch时,如果等待线程被阻塞,同时它也负责某个countDown()调用,且这个调用依赖于其他线程的动作,可能导致死锁。

解决方案:确保countDown()调用不会被阻塞,或者在设计时避免让等待await()的线程也负责减少计数器。可以通过分离职责或使用其他同步工具(如SemaphoreCyclicBarrier)来避免此类死锁。

2. 计数器设置错误

问题描述:初始化CountDownLatch时,计数器设置错误,导致等待线程提前或永不释放。

解决方案:仔细校验和计算初始计数值,确保它准确反映了需要等待的事件数量。在复杂场景中,可以使用动态计数器(如通过AtomicInteger管理)并在所有任务启动前确定最终计数值。

3. 资源泄漏

问题描述:如果使用不当,如在等待线程中没有正确处理异常,可能导致资源泄漏,如线程池中的线程无法正常回收。

解决方案:在await()调用中捕获所有异常,并确保在异常情况下也能调用countDown()或释放其他共享资源。使用try-with-resources或finally块确保资源的清理。

4. 过度阻塞

问题描述:大量线程调用await()等待,可能会导致CPU资源浪费在上下文切换上,影响性能。

解决方案:尽量减少等待线程的数量,或者优化任务执行逻辑,减少同步点。考虑使用更细粒度的并发控制机制,如SemaphoreConcurrentHashMap,以减少阻塞等待。

5. 调试困难

问题描述:在并发环境下,使用CountDownLatch可能导致程序行为难以预测和调试,特别是当涉及多个并发组件时。

解决方案:增强日志记录,记录每个线程的执行状态和CountDownLatch的关键操作(如计数器变化、线程等待和释放)。使用专业的并发分析工具(如VisualVM、JProfiler)来监控线程活动和锁的使用情况。

6. 中断处理不当

问题描述:调用await()的线程被中断,但未妥善处理中断信号,可能导致线程状态混乱或资源泄露。

解决方案:在await()调用中捕获InterruptedException,并根据应用逻辑决定是重新尝试等待还是退出等待逻辑。确保在处理中断时清理资源并恢复线程到安全状态。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深入理解JavaScript事件循环Event Loop:宏任务与微任务的奇幻之旅

&#x1f525; 个人主页&#xff1a;空白诗 文章目录 &#x1f389; 引言&#x1f31f; 什么是事件循环&#xff1f;&#x1f4da; 「宏任务」 vs 「微任务」「宏任务」(Macrotask)「微任务」(Microtask)实际应用中的注意事项 &#x1f500; 执行流程概览&#x1f4dd; 代码示例…

如何在JavaScript/Vue中获取当前时间并格式化输出(精确到时分秒)

如何在JavaScript/Vue中获取当前时间并格式化输出&#xff08;精确到时分秒&#xff09; 不只是树&#xff0c;人也是一样&#xff0c;在不确定中生活的人&#xff0c;能比较经得起生活的考验&#xff0c;会锻炼出一颗独立自主的心。在不确定中&#xff0c;就能学会把很少的养分…

在uniapp中如何安装axios并解决跨域问题

目录 1、安装axios 2、导入 3、使用&#xff08;发请求&#xff09; 2.解决跨域问题 1.为什么要解决跨域问题&#xff1f; 2.前端如何解决跨域问题&#xff1f; 1、安装axios npm install axios 2、导入 在main.js中导入使用 import axios from axios; // 创建一个名…

智慧互联,统信UOS V20桌面专业版(1070)解锁办公新模式丨年度更新

从小屏到大屏 突破&#xff0c;就在方寸之间 从人机到智脑 融合&#xff0c;旨在新质生产力 统信UOS一直致力于将先进科技与用户场景相结合&#xff0c;不断提升用户的工作效率和生产力。在最新发布的统信UOS V20桌面专业版&#xff08;1070&#xff09;版本中&#xff0c;我们…

Rust 解决循环引用

导航 循环引用一、现象二、解决 循环引用 循环引用出现的一个场景就是你指向我&#xff0c;我指向你&#xff0c;导致程序崩溃 解决方式可以通过弱指针&#xff0c;而Rust中的弱指针就是Weak 在Rc中&#xff0c;可以实现&#xff0c;对一个变量&#xff0c;持有多个不可变引…

【一支射频电缆的诞生】GORE 戈尔

工具连接&#xff1a; https://microwave-cablebuilder.gore.com/ 控制参数&#xff1a; 连接器&#xff1a; 欣赏

Android APP读写外置SD卡无权限 java.io.IOException: Permission denied

在物联网应用里&#xff0c;app需要对挂载SD卡读写文件&#xff0c;从 Android 4.4&#xff08;KitKat&#xff09;版本开始&#xff0c;Google 引入了一项名为 "Storage Access Framework" 的新功能&#xff0c;该功能限制了应用对外部存储的直接读写权限,要不然就是…

【挑战30天首通《谷粒商城》】-【第一天】03、简介-分布式基础概念

文章目录 课程介绍 ( 本章了解即可&#xff0c;可以略过)1、微服务简而言之: 2、集群&分布式&节点2.1、定义2.2、示例 3、远程调用4、负载均衡常见的负裁均衡算法: 5、服务注册/发现&注册中心6、配置中心7、服务熔断&服务降级7.1、服务熔断7.2、服务降级 8、AP…

Appium测试之获取appPackage和appActivity

appPackage和appActivity 进行appium自动化测试非常重要的两个参数&#xff0c;我们所测试的APP不同&#xff0c;这两个参数肯定也是不一样的。那如何快速的获取这APP的这两个参数呢&#xff1f;我这里介绍两个方法。 import org.openqa.selenium.remote.DesiredCapabilities;i…

从零开始学AI绘画,万字Stable Diffusion终极教程(六)

【第6期】知识补充 欢迎来到SD的终极教程&#xff0c;这是我们的第六节课&#xff0c;也是最后一节课 这套课程分为六节课&#xff0c;会系统性的介绍sd的全部功能&#xff0c;让你打下坚实牢靠的基础 1.SD入门 2.关键词 3.Lora模型 4.图生图 5.controlnet 6.知识补充 …

利用生成式AI重新构想ITSM的未来

对注入 AI 的生成式 ITSM 的需求&#xff0c;在 2023 年 Gartner AI 炒作周期中&#xff0c;生成式 AI 达到预期值达到顶峰后&#xff0c;三分之二的企业已经将生成式 AI 集成到其流程中。 你问为什么这种追求&#xff1f;在预定义算法的驱动下&#xff0c;IT 服务交付和管理中…

PyCharm 集成 Git

目录 1、配置 Git 忽略文件 2、定位Git 3、使用pycharm本地提交 3.1、初始化本地库 3.2、添加到暂存区 3.3、提交到本地库 3.4、切换版本 4、分支操作 4.1、创建分支 4.2、切换分支 4.3、合并分支 5、解决冲突 1、配置 Git 忽略文件 作用&#xff1a;与项目的实际…

【redis】Redis五种常用数据类型和内部编码,以及对String字符串类型的总结

˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱ ʕ̯•͡˔•̯᷅ʔ大家好&#xff0c;我是xiaoxie.希望你看完之后,有不足之处请多多谅解&#xff0c;让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ აxiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客 本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN 如…

51单片机入门:蜂鸣器

蜂鸣器介绍 蜂鸣器是一种将电信号转换为声音信号的器件&#xff0c;常用来产生设备的按键音、报警音等提示信号。 蜂鸣器的种类 1、从结构上&#xff1a;压电式蜂鸣器和电磁式蜂鸣器。 压电式蜂鸣器&#xff1a;通过压电陶瓷的压电效应原理工作的。当加有交变电压时&#xf…

正点原子[第二期]Linux之ARM(MX6U)裸机篇学习笔记-14-主频和时钟配置

前言&#xff1a; 本文是根据哔哩哔哩网站上“正点原子[第二期]Linux之ARM&#xff08;MX6U&#xff09;裸机篇”视频的学习笔记&#xff0c;在这里会记录下正点原子 I.MX6ULL 开发板的配套视频教程所作的实验和学习笔记内容。本文大量引用了正点原子教学视频和链接中的内容。…

轮式机器人简介

迄今为止,轮子一般是移动机器人学和人造交通车辆中最流行的运动机构。它可达到很高的效率, 如图所示, 而且用比较简单的机械就可实现它的制作。 另外,在轮式机器人设计中,平衡通常不是一个研究问题。 因为在所有时间里,轮式机器人一般都被设计成在任何时间里所有轮子均与地接…

idea上如何新建git分支

当前项目在dev分支&#xff0c;如果想在新分支上开发代码&#xff0c;如何新建一个分支呢&#xff1f;5秒搞定~ 1、工具类选择git&#xff0c;点击New Branch 或者右下角点击git分支&#xff0c;再点击New Branch 2、在弹出的Create New Branch弹窗中&#xff0c;输入你的新分支…

VMware 不能拍摄快照

问题&#xff1a; 拍摄快照后&#xff0c;会出现这个弹窗&#xff0c;然后虚拟机就直接自动退出了&#xff0c;还会弹出一个框&#xff1a; 解决方法&#xff1a; 我用的是 window11 和 VMware16.0.0 这是因为VM16与window11&#xff0c;二者之间版本不兼容问题&#xff0c;可…

Python 全栈系列243 S2S flask_celery

说明 按现有的几个架构部件&#xff0c;构建数据流。 S Redis Stream。这个可以作为缓冲队列和简单任务队列&#xff0c;速度非常快&#xff0c;至少是万条/秒的速度。 Q RabbitMQ。这个作为任务队列&#xff0c;消息也主要是元数据。读速比较慢&#xff0c;但有一些特性&a…

Linux(openEuler、CentOS8)常用的IP修改方式(文本配置工具nmtui+配置文件+nmcli命令)

----本实验环境为openEuler系统<以server方式安装>&#xff08;CentOS类似&#xff0c;可参考本文&#xff09;---- 一、知识点 &#xff08;一&#xff09;文本配置工具nmtui(openEuler已预装) nmtui&#xff08;NetworkManager Text User Interface&#xff09;是一…