CountDownLatch通过计数器实现线程同步,初始化指定计数值,调用countDown()使计数减1,await()使线程阻塞直至计数为0;常用于多个线程同时启动或分阶段启动场景,如主线程等待工作线程准备就绪后再统一发令,或让某些线程依赖其他线程完成初始化后再执行;示例中使用一个CountDownLatch控制三个线程在准备完成后同时开始执行,输出显示它们几乎同时打印“开始执行任务”;另一场景用两个线程初始化完成后触发第三个线程启动,通过initComplete.await()实现依赖控制;使用时需注意初始值设置正确、确保每个事件都调用countDown()、避免重复使用实例、妥善处理中断异常,并在必要时选择CyclicBarrier替代。

在Java多线程编程中,经常需要控制多个线程的执行顺序,比如让某些线程等待其他线程完成初始化后再开始执行。CountDownLatch 是 java.util.concurrent 包中一个非常实用的同步工具类,它通过一个计数器实现线程间的协调,非常适合用于控制线程启动顺序。
CountDownLatch 内部维护一个计数器,初始化时指定计数值。当调用 countDown() 方法时,计数器减1;而其他线程可以调用 await() 方法阻塞等待,直到计数器变为0才会继续执行。这个机制天然适合用来做“等待一组操作完成”或“控制启动时机”的场景。
关键点:假设我们有三个线程:ThreadA、ThreadB 和 ThreadC,希望它们都准备好后同时启动。我们可以使用一个 CountDownLatch 实例作为“起跑枪”,确保所有线程在同一起点出发。
示例代码:
立即学习“Java免费学习笔记(深入)”;
public class StartTogetherExample {
public static void main(String[] args) throws InterruptedException {
CountDownLatch readyLatch = new CountDownLatch(1); // 控制启动信号
for (int i = 1; i <= 3; i++) {
final int threadId = i;
new Thread(() -> {
System.out.println("线程" + threadId + "已准备就绪,等待启动信号...");
try {
readyLatch.await(); // 所有线程在此等待
System.out.println("线程" + threadId + "开始执行任务");
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
}).start();
}
// 主线程模拟准备工作耗时
Thread.sleep(2000);
System.out.println("所有线程已准备完毕,发出启动信号!");
readyLatch.countDown(); // 计数器减为0,释放所有等待线程
}
}
输出结果会显示:三个线程几乎在同一时间打印“开始执行任务”,说明它们被成功同步启动。
有时候不仅需要“同时启动”,还可能要求“按阶段启动”。例如,先让前两个线程完成初始化,再通知第三个线程启动。这时可以用多个 CountDownLatch 配合完成。
示例:分阶段启动
CountDownLatch initComplete = new CountDownLatch(2); // 等待前两个线程初始化完成
Thread t1 = new Thread(() -> {
System.out.println("T1 正在初始化...");
try { Thread.sleep(1000); } catch (InterruptedException e) {}
System.out.println("T1 初始化完成");
initComplete.countDown();
});
Thread t2 = new Thread(() -> {
System.out.println("T2 正在初始化...");
try { Thread.sleep(1500); } catch (InterruptedException e) {}
System.out.println("T2 初始化完成");
initComplete.countDown();
});
Thread t3 = new Thread(() -> {
System.out.println("T3 等待 T1 和 T2 初始化完成...");
try {
initComplete.await();
System.out.println("T3 开始执行");
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
});
t1.start(); t2.start(); t3.start();
在这个例子中,t3 必须等到 t1 和 t2 都调用了 countDown() 后才能继续执行,实现了线程间的依赖控制。
为了正确高效地使用 CountDownLatch 控制线程顺序,注意以下几点:
以上就是在Java中如何使用CountDownLatch控制线程启动顺序_CountDownLatch线程顺序控制技巧解析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号