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

CountDownLatch入门代码解析

文章目录

      • 核心思想:火箭发射倒计时 🚀
      • 最简单易懂的代码示例
      • 代码解析
      • 运行流程分析

核心思想:火箭发射倒计时 🚀

想象一下发射火箭的场景,在按下最终的发射按钮之前,必须有好几个系统同时完成自检,比如:

  1. 燃料系统检查
  2. 引擎系统检查
  3. 导航系统检查

控制中心(主线程)必须等待这3个检查全部报告“正常”后,才能下达“发射”指令。

CountDownLatch 就好比是这个场景中的倒计时计数器

  • CountDownLatch latch = new CountDownLatch(3);

    • 这等于在控制中心设置了一个初始值为 3 的倒计时器。意味着我们需要等待3个检查任务完成。
  • latch.await(); (等待)

    • 控制中心(主线程)调用这个方法,然后就进入等待状态。它会一直在这里被阻塞,直到倒计时器的数字变成 0
  • latch.countDown(); (倒数)

    • 每个检查系统(工作线程)在完成自己的任务后,就调用一次这个方法。
    • 每调用一次,倒计时器的数字就减一
    • 当第三个检查系统也调用了 countDown() 后,倒计时器数字变为0,await() 的等待结束,控制中心(主线程)被唤醒,继续执行后续的发射指令。

最简单易懂的代码示例

下面我们就用代码来模拟这个“火箭发射”的场景。

package CoountDownLatch;
import java.util.Random;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class SimpleCountDownLatchDemo {public static void main(String[] args) throws InterruptedException {// 1. 创建一个 CountDownLatch,计数器设置为 3//    意味着我们需要等待3个任务完成final CountDownLatch latch = new CountDownLatch(3);// 创建一个线程池来管理我们的检查任务ExecutorService executor = Executors.newFixedThreadPool(3);System.out.println("主控室:准备发射火箭,等待各系统检查...");// 2. 分配3个检查任务给不同的线程for (int i = 1; i <= 3; i++) {final String checkerName = "检查员-" + i;executor.submit(() -> {try {System.out.println("--> [" + checkerName + "] 开始进行系统检查...");// 模拟检查耗时Thread.sleep(new Random().nextInt(2000) + 1000); // 随机耗时1-3秒System.out.println("... [" + checkerName + "] 检查完成,已报告!");} catch (InterruptedException e) {e.printStackTrace();} finally {// 3. 关键!任务完成,调用 countDown(),计数器减一latch.countDown();}});}// 4. 主线程调用 await() 进入等待//    它会一直阻塞在这里,直到 latch 的计数器变为 0System.out.println("主控室:所有检查任务已派出,等待报告...");latch.await();// --- 当所有检查任务都调用了 countDown() 后,主线程才会从 await() 返回,执行以下代码 ---System.out.println("主控室:所有系统检查完成!准备发射!");System.out.println("3... 2... 1... 火箭发射!🚀");// 关闭线程池executor.shutdown();}
}

代码解析

  1. new CountDownLatch(3): 设置了一个需要3个“报告”才能继续的门闩。
  2. executor.submit(...): 我们派出了3个检查员(线程)去并行工作。
  3. latch.countDown(): 这是每个检查员完成工作后必须要做的事——向控制中心报告“我搞定了”。每报告一次,倒计时就减一。
  4. latch.await(): 这是主线程(控制中心)的等待点。它会一直卡在这里,直到收到全部3个“搞定了”的报告。

运行流程分析

  1. 程序启动,main 线程打印 “准备发射火箭…”。
  2. 3个检查员线程被创建并开始并行地执行检查(你会看到3条 “开始进行系统检查…” 的日志)。
  3. main 线程打印 “所有检查任务已派出…” 后,立刻调用 latch.await()进入阻塞等待
  4. 在接下来的几秒内,你会看到检查员们随机地、不按顺序地完成他们的工作,并打印 “检查完成,已报告!”。每完成一个,latch 的计数就减一。
  5. 第三个检查员也完成并调用 countDown() 后,latch 的计数变为0。
  6. main 线程的 await() 立刻被唤醒,程序继续执行,打印出最终的 “火箭发射!🚀”。

这个模式非常适合一个主线程需要等待多个子任务全部执行完毕后再进行汇总或执行下一步的场景。

流程:

  1. 定义Latch数量
  2. 在多线程任务中每次完成就latch.countDown();
  3. 在主线程中调用latch.await();进入等待,它会一直阻塞在这里,直到 latch 的计数器变为 0
  4. 当所有检查任务都调用了 countDown() 后,主线程才会从 await() 返回

相关文章:

  • ELK日志文件分析系统——L(Logstash)
  • Flask 动态模块注册
  • python中的异常处理try-except - else - finally与自定义异常处理
  • 探索数据的力量:Elasticsearch中指定链表字段的统计查询记录
  • 生日悖论理论及在哈希函数碰撞中的应用
  • AndroidMJ-mvp与mvvm
  • ASR语音转写技术全景解析:从原理到实战
  • 人工智能学习21-Pandas-pivot_table
  • 关于MCU、MPU、SoC、DSP四大类型芯片
  • 基于区块链的去中心化身份验证系统:原理、实现与应用
  • 【软测】接口测试 - 用postman测试软件登录模块
  • GDI+ 中与GDI32取图形区域函数对比CreateEllipticRgn/CreatePolygonRgn
  • day31 打卡
  • 茶文化部分答案
  • 数据库学习(六)——MySQL事务
  • Linux文件权限管理核心要点总结
  • 【图片转 3D 模型】北大·字节跳动·CMU携手——单图15 秒生成结构化3D模型!
  • 精准测量 MySQL 主从复制延迟—pt-heartbeat工具工作原理
  • 8088单板机8259中断的软件触发测试
  • Python全栈开发:前后端分离项目架构详解
  • 华威桥网站建设/seopeix
  • 上海微网站建设/网络营销学什么
  • 做电商必须知道的网站/app运营推广是干什么
  • 邢台疫情最新消息2021/虞城seo代理地址
  • 有没有做网站的/如何进行网站的推广
  • 如何建设网站 知乎/网上做推广怎么收费