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.mzph.cn/bicheng/10110.shtml

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

相关文章

uniapp——弹出键盘遮挡住输入框 textarea,处理方法

案例 在写输入框的时候会遇见 键盘遮挡住部分textarea框的一部分&#xff0c;使用cursor-spacing处理即可 修改后&#xff1a; 其他问题&#xff1a; 调起键盘输入时&#xff0c;不希望上方的内容被顶上去 代码 <view class"commentBox" :style"botto…

爬虫学习(4)每日一笑

代码 import requests import re import osif __name__ "__main__":if not os.path.exists("./haha"):os.makedirs(./haha)url https://mlol.qt.qq.com/go/mlol_news/varcache_article?docid6321992422382570537&gameid3&zoneplat&webview…

Linux 认识与学习Bash——3

在Linux bash中&#xff0c;数据流重定向是指将命令的输出从默认的标准输出&#xff08;通常是终端&#xff09;重定向到其他位置&#xff0c;如文件或另一个命令的输入。这是通过使用特定的符号来实现的。例如&#xff0c;>用于将输出重定向到文件&#xff0c;而<用于将…

Proxmox VE 8 用SDN隔离用户网络

作者&#xff1a;田逸&#xff08;formyz&#xff09; 最新发布的Proxmox VE&#xff08;以下简称PVE&#xff09; 8在Web管理后台集成了易于操作的SDN&#xff08;软件定义网络&#xff09;功能插件&#xff0c;其实质是对不同的PVE用户指定不同的网络&#xff0c;进行逻辑隔离…

[移动通讯]【无线感知-P1】[从菲涅尔区模型到CSI模型-3][Mobius transformations-7】【Inversion】

前言&#xff1a; mobius map 里面比较难的是inversion &#xff0c;林菲尔德学院&#xff08;Linfield College&#xff09; Michael P. Hitchman. 有本书详细介绍一下该方向的一些原理&#xff0c;例子. Whitman College Book: 《Geometry with an Introduction to Cosmic T…

el-select选项框内容过长

利用popper-class实现选项框内容过长&#xff0c;截取显示功能&#xff1a; <el-select popper-class"popper-class" :popper-append-to-body"false" v-model"value" placeholder"请选择"><el-optionv-for"item in opt…

初识 Linux线程

再学习完Linux进程后,本期,我们来讲解Linux线程 1.为什么需要线程 在之前学习进程前,我们写的所有代码几乎都是单个执行流的,也就是说我们的代码只有一条路走. 在学习进程后,我们可以通过fork进行进程创建,给进程分配任务进行多执行流执行任务,问题来了 那我们为什么还需要…

清空回收站是彻底删除吗?一文解答你的疑问!

“我刚刚本来想在回收站中恢复一个文件的&#xff0c;但是一不小心就清空了回收站&#xff0c;想问问清空回收站是彻底删除吗&#xff1f;清空了回收站文件还有机会找回吗&#xff1f;” 在使用电脑的过程中&#xff0c;我们经常会将不再需要的文件或文件夹移动到回收站&#x…

数据结构与算法学习笔记六-二叉树的顺序存储表示法和实现(C语言)

目录 前言 1.数组和结构体相关的一些知识 1.数组 2.结构体数组 3.递归遍历数组 2.二叉树的顺序存储表示法和实现 1.定义 2.初始化 3.先序遍历二叉树 4.中序遍历二叉树 5.后序遍历二叉树 6.完整代码 前言 二叉树的非递归的表示和实现。 1.数组和结构体相关的一些知…

AUTOSAR OS调度表讲解

调度表 AUTOSAR OS通过调度表(Schedule Table)来解决一个alarm只能激活一个任务的限制。调度表是预定义的行为序列,通过到期点实现。AUTOSAR OS遍历调度表并依次处理每个到期点,遍历由底层的counter来实现驱动。 到期点发生在从概念零开始的静态配置偏移量上。偏移量在静…

【程序设计和c语言-谭浩强配套】(适合专升本、考研)

一晃大半年没更新了&#xff0c;这一年一直在备考&#xff0c;想着这几天把前段时间学的c语言给大家分享一下&#xff0c;在此做了一个专栏&#xff0c;有需要的小伙伴可私信获取o。 简介&#xff1a;本专栏所有内容皆适合专升本、考研的复习资料&#xff0c;本人手上也有日常…

【2024亚马逊云科技峰会】Amazon Bedrock + Llama3 生成式AI实践

在 4 月 18 日&#xff0c;Meta在官网上公布了旗下最新大模型Llama 3。目前&#xff0c;Llama 3已经开放了80亿&#xff08;8B&#xff09;和700亿&#xff08;70B&#xff09;两个小参数版本&#xff0c;上下文窗口为8k&#xff0c;据称&#xff0c;通过使用更高质量的训练数据…

【基础算法总结】二分查找一

二分查找一 1. 二分查找2.在排序数组中查找元素的第一个和最后一个位置3.x 的平方根4.搜索插入位置 点赞&#x1f44d;&#x1f44d;收藏&#x1f31f;&#x1f31f;关注&#x1f496;&#x1f496; 你的支持是对我最大的鼓励&#xff0c;我们一起努力吧!&#x1f603;&#x1…

制造业数字化转型解决方案及应用(125页PPT)

一、资料介绍 《制造业数字化转型解决方案及应用》是一份内容丰富、深入剖析制造业数字化转型的125页PPT资料。这份资料以“智能制造、制造业数字化转型、制造业数字化转型案例”为关键词&#xff0c;全面展现了制造业数字化转型的核心理念、解决方案以及实际应用案例。 关注…

【JS红宝书学习笔记】第1、2章

第1章 什么是JavaScript JavaScript 是一门用来与网页交互的脚本语言&#xff0c;包含以下三个组成部分。 ECMAScript&#xff1a;由 ECMA-262 定义并提供核心功能。文档对象模型&#xff08;DOM&#xff09;&#xff1a;提供与网页内容交互的方法和接口。浏览器对象模型&…

块元素、内联元素、行内块元素

一、介绍&#xff1a; CSS元素划分成块元素、行内元素&#xff08;内联元素&#xff09;、行内块元素等多种常用类型。也就是说&#xff1a;在CSS中&#xff0c;元素根据其在页面上的布局方式被分为不同的显示类型。 背景&#xff1a;HTML负责定义网页的结构和内容&#xff0c…

OC foudation框架(上)学习

foundation框架 文章目录 foundation框架字符串&#xff08;NSString && NSMutableString&#xff09;NSString的其他功能NSMutableString 日期与时间 &#xff08;NSDate&#xff09;2.1 日期与时间&#xff08;NSDate&#xff09;2.2日期格式器日历与日期组件定时器&…

Java入门基础学习笔记8——注释

1、注释&#xff1a; 注释是写在程序中对代码进行解释说明的文件&#xff0c;方便自己和其他人查看&#xff0c;以便理解程序的。 package cn.ensource.note;/**文档注释文档注释 */ public class NoteDemo {public static void main(String[] args) {// 单行注释System.out.…

word转pdf的java实现(documents4j)

一、多余的话 java实现word转pdf可用的jar包不多&#xff0c;很多都是收费的。最近发现com.documents4j挺好用的&#xff0c;它支持在本机转换&#xff0c;也支持远程服务转换。但它依赖于微软的office。电脑需要安装office才能转换。鉴于没在linux中使用office&#xff0c;本…

SwiftUI 调整视图内容周围间隙(Content Margins)的“时髦”方法

概述 在 SwiftUI 开发的应用中,往往在小屏设备(比如 iPhone)上布局良好的 App 放到大屏(iPad)上后就会“一塌糊涂”。因为它们一味的只想着“占据”却不知道“舍弃”。 从 iOS 17.0(iPad 17.0)开始苹果提供了原生的视图修改器方法专注于处理此事。 在本篇博文中,您将…