同步工具类之闭锁-CountDownLatch

本文详细介绍了闭锁的概念及其在多线程编程中的作用,特别聚焦于CountDownLatch的灵活实现方式。阐述了如何利用闭锁确保特定活动在其他活动完成后执行,同时提供了一个示例代码来展示闭锁的使用场景。

 

闭锁是一种同步工具类,可以延迟线程的进度直到其到达终止状态。闭锁的作用相当于一扇门:在闭锁到达结束状态之前,这扇门一直是关闭的,并且没有

任何线程能通过,当到达结束状态时,这扇门会打开并允许所有的线程通过。当闭锁到达结束状态后,将不会再改变状态,因此这扇门将永远保持打开状态。

闭锁可以用来确保某些活动直到其他活动都完成后才继续执行。

 

CountDownLatch是一种灵活的闭锁实现,可以在上述各种情况中使用,它可以使一个或多个线程等待一组事件发生。闭锁状态包括一个计数器,该计数器

被初始化为一个正数,表示需要等待的事件数量。countDown方法递减计数器,表示有一个事件已经发生了,而await方法等待计数器达到零,这表示所有

需要等待的事件都已经发生。如果计数器的值非零,那么await会一直阻塞直到计数器为零,或者等待中的线程中断,或者等待超时。

 

public class TestHarness {
    public long timeTasks(int nthreads, final Runnable task) throws InterruptedException {
        final CountDownLatch startGate = new CountDownLatch(1);
        final CountDownLatch endGate = new CountDownLatch(nthreads);

        for (int i = 0; i < nthreads; i++) {
            Thread t = new Thread(){
                @Override
                public void run() {
                    try{
                        startGate.await();
                        try{
                            task.run();
                        }finally {
                            endGate.countDown();
                        }
                    } catch (InterruptedException ignored) {}
                }
            };
            t.start();
        }

        long start = System.nanoTime();
        startGate.countDown();
        endGate.await();
        long end = System.nanoTime();
        return end-start;
    }
}

  

 

startGate计数器的初始值为1,而endGate计数器的初始值为工作线程的数量。每个工作线程首先要做的就是在startGate上等待,从而确保所有线程都就绪后

才开始执行。而每个线程要做的最后一件事情是将调用endGatecountDown方法减1,这能使主线程高效地等待直到所有工作线程都执行完成,因此可以更准确

地统计所消耗的时间

 

ps. 为什么要在上例中使用闭锁,而不是在线程创建后就立即启动?或许,我们希望测试n个线程并发执行某个任务时需要的时间。如果在创建线程后立即启动

它们,那么先启动的线程将“领先”后启动的线程,并且活跃线程数量会随着时间的推移而增加或减少,竞争程度也在不断发生变化。

 

### 介绍 CountDownLatch本质上是一种同步工具类,具有不同状态,可协调线程的控制流,以达到线程安全的目的。它的内部机制是阻塞一部分线程,使其在达到某个条件之后再执行。常见应用场景是开启多个线程同时执行某个任务,等所有任务执行完再统计汇总结果[^1][^2]。 ### 使用方法 其使用方式可参考ReentrantLock,开启多个线程同时执行任务,最后等待所有任务完成进行汇总。查看它的内部实现,设计模型和ReentrantLock类似,都是内部定义了一个Sync继承了AQS同步器。不过CountDownLatch内部方法不多。在构造方法上,若了解过ReentrantLock的实现,会比较熟悉。需要注意的是,CountDownLatch并没有实现Lock接口,因为它本质上是一个计数器,不需要获取锁和释放锁等操作。调用await()方法的线程将进入等待状态,直到count变为 0 [^2][^3][^4]。 ### 原理 CountDownLatch底层依赖AQS(AbstractQueuedSynchronizer)实现,利用ReentrantLock来进行线程间的同步和计数管理。线程调用await()方法进入等待状态,直到计数器count降为 0 时,被阻塞的线程才会继续执行 [^4]。 ### 示例代码 ```java import java.util.concurrent.CountDownLatch; public class CountDownLatchExample { public static void main(String[] args) throws InterruptedException { int threadCount = 3; CountDownLatch latch = new CountDownLatch(threadCount); for (int i = 0; i < threadCount; i++) { final int index = i; new Thread(() -> { try { System.out.println("Thread " + index + " is working."); Thread.sleep(1000); // 模拟任务执行 System.out.println("Thread " + index + " has finished."); } catch (InterruptedException e) { e.printStackTrace(); } finally { latch.countDown(); } }).start(); } // 主线程等待所有任务完成 latch.await(); System.out.println("All tasks have been completed."); } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值