CountDownLatch 详细介绍

CountDownLatch 是 Java 中 java.util.concurrent 包提供的一个同步工具类,用于协调多个线程之间的执行顺序。它允许一个或多个线程等待,直到其他线程完成一组操作后继续执行。CountDownLatch 是一种倒计数锁存器,通过设置一个初始计数器值,线程可以通过调用 countDown() 方法递减计数器,当计数器达到 0 时,等待的线程会被唤醒。

主要特点
  1. 一次性使用CountDownLatch 是一次性工具,一旦计数器减到 0,它不能被重置或重用。如果需要可重置的机制,可以考虑使用 CyclicBarrier
  2. 线程等待:一个或多个线程可以通过 await() 方法等待计数器归零。
  3. 计数器递减:通过 countDown() 方法,线程可以减少计数器的值。
  4. 线程安全CountDownLatch 是线程安全的,适合多线程并发场景。
  5. 灵活性:可以用于等待一组任务完成后再执行后续逻辑,或者让主线程等待多个子线程完成。
工作原理
  • CountDownLatch 内部维护一个计数器,初始化时指定一个正整数值。
  • 调用 countDown() 方法会将计数器减 1,直到计数器为 0。
  • 调用 await() 方法的线程会阻塞,直到计数器为 0,所有等待的线程会被唤醒。
  • 使用基于 AQS(AbstractQueuedSynchronizer)的同步机制,确保线程安全和高效的等待/通知机制。
常见方法
  • CountDownLatch(int count):构造方法,初始化计数器为指定的值 count
  • void countDown():将计数器减 1,如果计数器达到 0,唤醒所有等待的线程。
  • void await():使当前线程阻塞,直到计数器归零。
  • boolean await(long timeout, TimeUnit unit):使当前线程阻塞,直到计数器归零或超时,返回是否成功(true 表示计数器归零,false 表示超时)。
  • long getCount():返回当前计数器的值。

使用场景

  1. 等待多个线程完成:主线程等待多个子线程完成任务后再继续执行。
  2. 并行任务协调:在任务分解为多个子任务时,确保所有子任务完成后再进行汇总。
  3. 启动信号:确保多个线程同时开始执行(通过设置计数器为 1,主线程调用 countDown() 触发所有线程)。
  4. 测试并发场景:在性能测试中,模拟多个线程同时执行某个任务。

CountDownLatch 示例代码

以下是几个使用 CountDownLatch 的实际例子,展示其在不同场景下的应用。

示例 1:主线程等待多个子线程完成
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class CountDownLatchExample1 {
    public static void main(String[] args) throws InterruptedException {
        int threadCount = 3;
        // 创建一个计数器,初始值为3
        CountDownLatch latch = new CountDownLatch(threadCount);
        
        // 创建线程池
        ExecutorService executor = Executors.newFixedThreadPool(threadCount);
        
        // 提交三个任务
        for (int i = 1; i <= threadCount; i++) {
            final int taskId = i;
            executor.submit(() -> {
                try {
                    System.out.println("任务 " + taskId + " 开始执行");
                    Thread.sleep((int) (Math.random() * 1000)); // 模拟任务耗时
                    System.out.println("任务 " + taskId + " 完成");
                    latch.countDown(); // 计数器减1
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            });
        }
        
        // 主线程等待所有任务完成
        System.out.println("主线程等待所有任务完成...");
        latch.await();
        System.out.println("所有任务已完成,主线程继续执行");
        
        // 关闭线程池
        executor.shutdown();
    }
}

输出示例

主线程等待所有任务完成...
任务 1 开始执行
任务 2 开始执行
任务 3 开始执行
任务 1 完成
任务 3 完成
任务 2 完成
所有任务已完成,主线程继续执行

解释

  • 创建一个 CountDownLatch,初始计数器为 3。
  • 三个线程分别执行任务,并在完成后调用 countDown() 减少计数器。
  • 主线程调用 await(),阻塞直到计数器为 0。
  • 当所有线程完成任务后,计数器归零,主线程继续执行。
示例 2:模拟并发任务启动

这个例子展示如何使用 CountDownLatch 让多个线程同时开始执行任务。

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class CountDownLatchExample2 {
    public static void main(String[] args) throws InterruptedException {
        int threadCount = 5;
        // 创建一个计数器,初始值为1
        CountDownLatch startSignal = new CountDownLatch(1);
        
        // 创建线程池
        ExecutorService executor = Executors.newFixedThreadPool(threadCount);
        
        // 提交5个任务,等待启动信号
        for (int i = 1; i <= threadCount; i++) {
            final int workerId = i;
            executor.submit(() -> {
                try {
                    System.out.println("工人 " + workerId + " 准备就绪");
                    startSignal.await(); // 等待启动信号
                    System.out.println("工人 " + workerId + " 开始工作");
                    Thread.sleep((int) (Math.random() * 1000)); // 模拟工作
                    System.out.println("工人 " + workerId + " 完成工作");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            });
        }
        
        // 主线程模拟准备时间
        Thread.sleep(1000);
        System.out.println("所有工人准备就绪,主线程发出启动信号!");
        startSignal.countDown(); // 发出启动信号
        
        // 关闭线程池
        executor.shutdown();
    }
}

输出示例

工人 1 准备就绪
工人 2 准备就绪
工人 3 准备就绪
工人 4 准备就绪
工人 5 准备就绪
所有工人准备就绪,主线程发出启动信号!
工人 1 开始工作
工人 2 开始工作
工人 3 开始工作
工人 4 开始工作
工人 5 开始工作
工人 3 完成工作
工人 1 完成工作
工人 5 完成工作
工人 2 完成工作
工人 4 完成工作

解释

  • 使用 CountDownLatch 的计数器为 1,模拟一个启动信号。
  • 所有工人线程调用 await() 等待主线程的信号。
  • 主线程调用 countDown() 后,所有工人线程几乎同时开始工作。
示例 3:带超时的等待

这个例子展示如何使用带超时的 await 方法。

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

public class CountDownLatchExample3 {
    public static void main(String[] args) throws InterruptedException {
        int threadCount = 2;
        CountDownLatch latch = new CountDownLatch(threadCount);
        
        ExecutorService executor = Executors.newFixedThreadPool(threadCount);
        
        // 提交两个任务
        executor.submit(() -> {
            try {
                System.out.println("任务 1 开始执行");
                Thread.sleep(1000); // 模拟耗时任务
                System.out.println("任务 1 完成");
                latch.countDown();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        
        executor.submit(() -> {
            try {
                System.out.println("任务 2 开始执行");
                Thread.sleep(3000); // 模拟更长的耗时任务
                System.out.println("任务 2 完成");
                latch.countDown();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        
        // 主线程等待,最多等待2秒
        System.out.println("主线程等待,最多2秒...");
        boolean completed = latch.await(2, TimeUnit.SECONDS);
        if (completed) {
            System.out.println("所有任务在2秒内完成");
        } else {
            System.out.println("超时,部分任务未完成");
        }
        
        executor.shutdown();
    }
}

输出示例

主线程等待,最多2秒...
任务 1 开始执行
任务 2 开始执行
任务 1 完成
超时,部分任务未完成
任务 2 完成

解释

  • 主线程设置了 2 秒的超时时间,通过 await(2, TimeUnit.SECONDS)
  • 任务 1 在 1 秒内完成,但任务 2 需要 3 秒,因此主线程在超时后继续执行,输出“部分任务未完成”。

CountDownLatch 与其他工具的对比

  1. 与 CyclicBarrier 的区别
    • CountDownLatch 是一次性的,计数器归零后无法重用;CyclicBarrier 可以重置并重复使用。
    • CountDownLatch 通常用于主线程等待子线程;CyclicBarrier 更适合多个线程相互等待。
  2. 与 Semaphore 的区别
    • Semaphore 用于控制资源访问的并发数;CountDownLatch 用于协调线程的完成顺序。
  3. 与 wait/notify
    • CountDownLatch 提供了更简单、更高级的同步机制,基于 AQS 实现,性能更优。

使用建议

  • 适用场景:当需要等待一组任务完成后再执行后续逻辑,或需要多个线程同时开始时。
  • 不适用场景:如果需要重复使用计数器,建议使用 CyclicBarrier;如果需要控制并发访问量,建议使用 Semaphore
  • 注意事项
    • 确保初始计数器值合理,过高的计数可能导致线程长时间等待。
    • countDown() 调用过多不会导致计数器变成负数,但应避免逻辑错误。
    • 使用带超时的 await 方法可以避免无限等待。

总结

CountDownLatch 是一个简单而强大的同步工具,适用于协调多线程任务的完成顺序或统一启动。通过设置计数器和使用 await/countDown 方法,它能有效地控制线程间的协作。上述示例展示了其在等待任务完成、统一启动并发任务和带超时等待的场景中的应用。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Ricky_Ribbon

你的鼓励将是我创作的最大动力!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值