Java之CountDownLatch使用

CountDownLatch作为Java并发工具包中的重要组件,用于实现一个或多个线程等待其他线程完成任务后再继续执行的场景。本文详细介绍了CountDownLatch的工作原理、使用方法及应用场景,并通过一个模拟赛跑的示例来演示其具体使用。

CountDownLatch

1、类介绍

一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待。用给定的计数 初始化 CountDownLatch。由于调用了 countDown() 方法,所以在当前计数到达零之前,await 方法会一直受阻塞。之后,会释放所有等待的线程,await 的所有后续调用都将立即返回。这种现象只出现一次——计数无法被重置。 一个线程(或者多个), 等待另外N个线程完成某个事情之后才能执行

2、使用场景
在一些应用场合中,需要等待某个条件达到要求后才能做后面的事情;同时当线程都完成后也会触发事件,以便进行后面的操作。 这个时候就可以使用CountDownLatch。CountDownLatch最重要的方法是countDown()和await(),前者主要是倒数一次,后者是等待倒数到0,如果没有到达0,就只有阻塞等待了。
3、方法说明

countDown

public void countDown()
递减锁存器的计数,如果计数到达零,则释放所有等待的线程。如果当前计数大于零,则将计数减少。如果新的计数为零,出于线程调度目的,将重新启用所有的等待线程。

如果当前计数等于零,则不发生任何操作。

await

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

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

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

如果计数到达零,则该方法返回 true 值。

如果当前线程:

  • 在进入此方法时已经设置了该线程的中断状态;或者
  • 在等待时被中断

则抛出 InterruptedException,并且清除当前线程的已中断状态。如果超出了指定的等待时间,则返回值为 false。如果该时间小于等于零,则此方法根本不会等待。

 

参数:
timeout - 要等待的最长时间
unit - timeout 参数的时间单位。
返回:
如果计数到达零,则返回 true;如果在计数到达零之前超过了等待时间,则返回 false
抛出:
InterruptedException - 如果当前线程在等待时被中断
4、相关实例
<span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">public</span> <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">class</span> CountDownLatchTest {

    <span class="bh_code_a_Java_comments" style="margin: 0px; padding: 0px; border: 0px; color: rgb(63, 127, 95); background-color: transparent;">// 模拟了100米赛跑,10名选手已经准备就绪,只等裁判一声令下。当所有人都到达终点时,比赛结束。</span>
    <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">public</span> <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">static</span> <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">void</span> main(String[] args) <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">throws</span> InterruptedException {

        <span class="bh_code_a_Java_comments" style="margin: 0px; padding: 0px; border: 0px; color: rgb(63, 127, 95); background-color: transparent;">// 开始的倒数锁 </span>
        <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">final</span> CountDownLatch begin = <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">new</span> CountDownLatch(1);  

        <span class="bh_code_a_Java_comments" style="margin: 0px; padding: 0px; border: 0px; color: rgb(63, 127, 95); background-color: transparent;">// 结束的倒数锁 </span>
        <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">final</span> CountDownLatch end = <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">new</span> CountDownLatch(10);  

        <span class="bh_code_a_Java_comments" style="margin: 0px; padding: 0px; border: 0px; color: rgb(63, 127, 95); background-color: transparent;">// 十名选手 </span>
        <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">final</span> ExecutorService exec = Executors.newFixedThreadPool(10);  

        <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">for</span> (<span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">int</span> index = 0; index < 10; index++) {
            <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">final</span> <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">int</span> NO = index + 1;  
            Runnable run = <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">new</span> Runnable() {
                <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">public</span> <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">void</span> run() {  
                    <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">try</span> {  
                        <span class="bh_code_a_Java_comments" style="margin: 0px; padding: 0px; border: 0px; color: rgb(63, 127, 95); background-color: transparent;">// 如果当前计数为零,则此方法立即返回。</span>
                        <span class="bh_code_a_Java_comments" style="margin: 0px; padding: 0px; border: 0px; color: rgb(63, 127, 95); background-color: transparent;">// 等待</span>
                        begin.await();  
                        Thread.sleep((<span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">long</span>) (Math.random() * 10000));  
                        System.out.println(<span class="bh_code_a_Java_string" style="margin: 0px; padding: 0px; border: 0px; color: rgb(42, 0, 255); background-color: transparent;">"No."</span> + NO + <span class="bh_code_a_Java_string" style="margin: 0px; padding: 0px; border: 0px; color: rgb(42, 0, 255); background-color: transparent;">" arrived"</span>);  
                    } <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">catch</span> (InterruptedException e) {  
                    } <span class="bh_code_a_Java_keywords" style="margin: 0px; padding: 0px; border: 0px; color: rgb(127, 0, 85); background-color: transparent;">finally</span> {  
                        <span class="bh_code_a_Java_comments" style="margin: 0px; padding: 0px; border: 0px; color: rgb(63, 127, 95); background-color: transparent;">// 每个选手到达终点时,end就减一</span>
                        end.countDown();
                    }  
                }  
            };  
            exec.submit(run);
        }  
        System.out.println(<span class="bh_code_a_Java_string" style="margin: 0px; padding: 0px; border: 0px; color: rgb(42, 0, 255); background-color: transparent;">"Game Start"</span>);  
        <span class="bh_code_a_Java_comments" style="margin: 0px; padding: 0px; border: 0px; color: rgb(63, 127, 95); background-color: transparent;">// begin减一,开始游戏</span>
        begin.countDown();  
        <span class="bh_code_a_Java_comments" style="margin: 0px; padding: 0px; border: 0px; color: rgb(63, 127, 95); background-color: transparent;">// 等待end变为0,即所有选手到达终点</span>
        end.await();  
        System.out.println(<span class="bh_code_a_Java_string" style="margin: 0px; padding: 0px; border: 0px; color: rgb(42, 0, 255); background-color: transparent;">"Game Over"</span>);  
        exec.shutdown();  
    }
}
5、输出结果
Game Start
No.9 arrived
No.6 arrived
No.8 arrived
No.7 arrived
No.10 arrived
No.1 arrived
No.5 arrived
No.4 arrived
No.2 arrived
No.3 arrived
Game Over
光伏储能虚拟同步发电机VSG并网仿真模型(Similink仿真实现)内容概要:本文档介绍了光伏储能虚拟同步发电机(VSG)并网仿真模型的Simulink实现方法,重点在于通过建立光伏储能系统与虚拟同步发电机相结合的仿真模型,模拟其在并网过程中的动态响应与控制特性。该模型借鉴了同步发电机的惯性和阻尼特性,提升了新能源并网系统的频率和电压支撑能力,增强了系统的稳定性与可控性。文档还提及相关电力系统仿真技术的应用,包括逆变器控制、储能配置、并网稳定性分析等,并提供了完整的Simulink仿真文件及技术支持资源链接,便于科研人员复现与二次开发。; 适合人群:电气工程、自动化、能源系统等相关专业的研究生、科研人员及从事新能源并网技术开发的工程师。; 使用场景及目标:①用于研究光伏储能系统在弱电网条件下的并网稳定性问题;②掌握虚拟同步发电机(VSG)控制策略的设计与仿真方法;③支持高水平论文(如EI/SCI)的模型复现与创新研究;④为微电网、智能电网中的分布式能源接入提供技术参考。; 阅读建议:建议结合提供的Simulink模型文件与文档说明逐步操作,重点关注VSG控制模块的参数设置与动态响应分析,同时可延伸学习文中提及的MPPT、储能管理、谐波分析等相关技术,以提升综合仿真能力。
CountDownLatch 是一个同步工具类,在多线程环境中常常用于控制线程的执行顺序或者等待某个事件的发生。它的主要作用是让一组线程等待某个事件的发生,当事件发生后,所有等待的线程都可以继续执行。 CountDownLatch使用步骤如下: 1. 创建一个 CountDownLatch 对象,指定需要等待的线程数。 2. 在需要等待的线程中,调用 CountDownLatch 的 await() 方法,等待事件的发生。 3. 当事件发生时,调用 CountDownLatch 的 countDown() 方法,通知等待的线程可以继续执行。 下面是一个简单的例子: ``` import java.util.concurrent.CountDownLatch; public class CountDownLatchDemo { public static void main(String[] args) throws InterruptedException { CountDownLatch latch = new CountDownLatch(3); Thread t1 = new Thread(new Task("Task1", latch)); Thread t2 = new Thread(new Task("Task2", latch)); Thread t3 = new Thread(new Task("Task3", latch)); t1.start(); t2.start(); t3.start(); latch.await(); // 等待所有任务执行完成 System.out.println("All tasks have been completed."); } static class Task implements Runnable { private String name; private CountDownLatch latch; public Task(String name, CountDownLatch latch) { this.name = name; this.latch = latch; } @Override public void run() { System.out.println(name + " is running..."); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(name + " has completed."); latch.countDown(); } } } ``` 在这个例子中,我们创建了一个 CountDownLatch 对象,并指定需要等待的线程数为 3。然后创建了 3 个线程,并让它们执行一个简单的任务:输出线程名字、等待 1 秒钟,然后输出线程名字并调用 countDown() 方法。最后在主线程中调用 await() 方法等待所有任务执行完成。当所有任务执行完成后,输出一条“所有任务已完成”的消息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值