一篇不到300字短文,带你彻底理解Java并发包中CountDownLatch

一篇不到300字短文,带你彻底理解Java并发包中CountDownLatch,第1张

一篇不到300字短文,带你彻底理解Java并发包中CountDownLatch 了解CountDownLatch

Thread.join()实现的是一个线程等待另一个线程结束。但是如果有时候一个线程只需要等待其他线程特定 *** 作完成,不需要等待这些线程终止。这时候可以通过Java并发包中的Condition条件变量来实现,但是Java并发包中提供了更加方便直接的工具类——
java.util.concurrent.CountDownLatch。

CountDownLatch可以实现一个(或者多个)线程等待其他线程完成一组特定 *** 作之后继续运行,这组 *** 作被称为先决 *** 作。其内部维护了一个用于表示未完成的先决 *** 作数量的计数器。CountDownLatch.countDown()每次被调用就会使相应实例的计数器值减1。CountDownLatch.await()用来实现等待 *** 作。当计数器值为0时CountDownLatch.await()为空 *** 作;当计数器值不为0时CountDownLatch.await()执行的线程会暂停执行。

CountDownLatch构造方法:public CountDownLatch(int count),这里的count值为计数器的初始化值。

当count减为0时,count值不会在变化,此时调用CountDownLatch.countDown()不会抛出异常,并且后续执行CountDownLatch.await()的线程不会被暂停,所以 CountDownLatch是一次性的,一个CountDownLatch实例只能实现一次等待唤醒。

CountDownLatch使用案例
public class CountDownLatchMain {

    public static void main(String[] args) throws InterruptedException {
        //参会人员一共3个,所以这里计数器设置为3。
        CountDownLatch countDownLatch = new CountDownLatch(3);
        //A 、B、C参会人员模拟签到 *** 作 每个人签到一次计数器减1
        new Person("A", countDownLatch).signIn();
        new Person("B", countDownLatch).signIn();
        new Person("C", countDownLatch).signIn();
        // A 、B、C参会人员所有人签到完成,计数器为0,await则继续执行后续抽奖
        countDownLatch.await();
        System.out.println("主持人:开始抽奖!");
    }

    //模拟参会人员
    static class Person extends Thread{

        private CountDownLatch countDownLatch;

        private String name;

        public Person(String name, CountDownLatch countDownLatch){
            this.countDownLatch = countDownLatch;
            this.name = name;
        }

        //模拟签到
        public void signIn(){
            new SignIn(name, countDownLatch).start();
        }
    }

    //模拟签到工作
    static class SignIn extends Thread{
        private CountDownLatch countDownLatch;

        private String name;

        public SignIn(String name, CountDownLatch countDownLatch){
            this.countDownLatch = countDownLatch;
            this.name = name;
        }

        @Override
        public void run() {
            System.out.println(name + "签到完成");
            countDownLatch.countDown();
        }
    }
}

运行结果

上述代码模拟了会场所有人员签到完成后再进行抽奖的场景,参会人员签到执行countDownLatch.countDown()用来表示签到完成。CountDownLatch计数器为0时,则主持人开始抽奖。

CountDownLatch扩展

CountDownLatch.await()另一个版本:

public boolean await(long timeout, TimeUnit unit)

这个版本的await方法允许指定一个超时时间,如果CountDownLatch实例await时间超过设置的超时时间,即使计数器不为0,被await的线程也会被唤醒。此方法返回的boolean值结果用来区分是否由于超时等待唤醒。

对于同一个CountDownLatch实例latch,latch.countDown()的执行线程在执行该方法之前所执行的任何内存 *** 作对等待线程在latch.await()调用返回之后的代码是可见而且有序的。

 

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/5676894.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-12-17
下一篇 2022-12-17

发表评论

登录后才能评论

评论列表(0条)

保存