JAVA同步锁CountDownLatch

CountDownLatch是一个同步辅助类,用于让一个或多个线程等待其他线程完成操作。它通过一个计数器初始化,在计数器归零时释放所有等待的线程。常见应用场景包括并发任务的同步启动和多线程结果的汇总合并。CountDownLatch基于AQS实现,countDown()方法减少计数,await()方法使线程等待。与Thread.join相比,CountDownLatch提供更灵活的控制。
摘要由CSDN通过智能技术生成

CountDownLatch介绍

CountDownLatch(闭锁)是一个同步协助类,允许一个或多个线程等待,直到其他线程完成
操作集。
CountDownLatch使用给定的计数值(count)初始化。 await方法会阻塞直到当前的计数值
(count)由于countDown方法的调用达到0,count为0之后所有等待的线程都会被释放,并且随后对await方法的调用都会立即返回。这是一个一次性现象 —— count不会被重置。 如果你需要一个重置count的版本,那么请考虑使用CyclicBarrier。

CountDownLatch的使用

构造器

常用方法

 // 调用 await() 方法的线程会被挂起,它会等待直到 count 值为 0 才继续执行 
public void await() throws InterruptedException { }; 
// 和 await() 类似,若等待 timeout 时长后,count 值还是没有变为 0,不再等待,继续 执行
public boolean await(long timeout, TimeUnit unit) throws InterruptedExceptio n { }; 
// 会将 count 减 1,直至为 0 
public void countDown() { };

 CountDownLatch应用场景

CountDownLatch一般用作多线程倒计时计数器,强制它们等待其他一组(CountDownLatch的初始化决定)任务执行完成。
CountDownLatch的两种使用场景:
  • 场景1:让多个线程等待
  • 场景2:让单个线程等待。

场景1 让多个线程等待:模拟并发,让并发线程一起执行

/**
 * 让多个线程等待:模拟并发,让并发线程一起执行
 */
public class CountDownLatchTest {
    public static void main(String[] args) throws InterruptedException {

        CountDownLatch countDownLatch = new CountDownLatch(1);
        for (int i = 0; i < 5; i++) {
            new Thread(() -> {
                try {
                    //准备完毕……运动员都阻塞在这,等待号令
                    countDownLatch.await();
                    String parter = "【" + Thread.currentThread().getName() + "】";
                    System.out.println(parter + "开始执行……");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }).start();
        }

        Thread.sleep(2000);// 裁判准备发令
        countDownLatch.countDown();// 发令枪:执行发令
    }
}
场景2 让单个线程等待:多个线程(任务)完成后,进行汇总合并
     很多时候,我们的并发任务,存在前后依赖关系;比如数据详情页需要同时调用多个接口获取数据,并发请求获取到数据后、需要进行结果合并;或者多个数据操作完成后,需要数据check;这其实都是:在多个线程(任务)完成后,进行汇总合并的场景。
CountDownLatch countDownLatch = new CountDownLatch(5);
for (int i = 0; i < 5; i++) {
    final int index = i;
    new Thread(() -> {
        try {
            Thread.sleep(1000 +
                    ThreadLocalRandom.current().nextInt(1000));
            System.out.println(Thread.currentThread().getName()
                    + " finish task" + index);

            countDownLatch.countDown();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }).start();
}



// 主线程在阻塞,当计数器==0,就唤醒主线程往下执行。
countDownLatch.await();
System.out.println("主线程:在所有任务运行完成后,进行结果汇总");

CountDownLatch实现原理

      底层基于 AbstractQueuedSynchronizer 实现,CountDownLatch 构造函数中指定的count直接赋给AQS的state; 每次countDown()则都是release(1)减1,最后减到0时unpark唤醒队列中所有线程;这一步是由最后一个执行countdown方法的线程执行的。
      而调用await()方法时,当前线程就会判断state属性是否为0,如果为0,则不阻塞当前线程继续往下执行;如果不为0,则将线程加入等待队列并挂起当前线程,直到某个线程将state属性置为0,其就会唤醒在await()方法中等待的线程。

CountDownLatch与Thread.join的区别

  • CountDownLatch的作用就是允许一个或多个线程等待其他线程完成操作,看起来有点类似join() 方法,但其提供了比 join() 更加灵活的API。
  • CountDownLatch可以手动控制在n个线程里调用n次countDown()方法使计数器进行减一操作,也可以在一个线程里调用n次执行减一操作。
  • join() 的实现原理是不停检查join线程是否存活,如果 join 线程存活则让当前线程永远等待。所以两者之间相对来说还是CountDownLatch使用起来较为灵活。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

季风泯灭的季节

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

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

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

打赏作者

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

抵扣说明:

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

余额充值