zoukankan      html  css  js  c++  java
  • CyclicBarrier和CountDownLatch的用法与区别

    前言

    CyclicBarrier和CountDownLatch这两个工具都是在java.util.concurrent包下,并且平时很多场景都会使用到。

    本文将会对两者进行分析,记录他们的用法和区别。

    CountDownLatch

    CountDownLatch是一个非常实用的多线程控制工具类,称之为“倒计时器”,它允许一个或多个线程一直等待,直到其他线程的操作执行完后再执行。

    CountDownLatch是通过一个计数器来实现的,计数器的初始值为线程的数量。每当一个线程完成了自己的任务后,计数器的值就会减1。当计数器值到达0时,它表示所有的线程已经完成了任务,然后在闭锁上等待的线程就可以恢复执行任务。

    特点

    只能一次性使用(不能reset);主线程阻塞;某个线程中断将永远到不了屏障点,所有线程都会一直等待。

    例子

      //创建初始化3个线程的线程池
        private ExecutorService                    threadPool     = Executors.newFixedThreadPool(3);
        //保存每个学生的平均成绩
        private ConcurrentHashMap<String, Integer> map            = new ConcurrentHashMap<>();
        private CountDownLatch                     countDownLatch = new CountDownLatch(3);
    
        private void count() {
            for (int i = 0; i < 3; i++) {
                threadPool.execute(() -> {
                    //计算每个学生的平均成绩,代码略()假设为60~100的随机数
                    int score = (int) (Math.random() * 40 + 60);
                    try {
                        Thread.sleep(Math.round(Math.random() * 1000));
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    map.put(Thread.currentThread().getName(), score);
                    System.out.println(Thread.currentThread().getName() + "同学的平均成绩为" + score);
                    countDownLatch.countDown();
                });
            }
            this.run();
            threadPool.shutdown();
        }
    
        @Override
        public void run() {
            try {
                countDownLatch.await();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            int result = 0;
            Set<String> set = map.keySet();
            for (String s : set) {
                result += map.get(s);
            }
            System.out.println("三人平均成绩为:" + (result / 3) + "分");
        }
    
        public static void main(String[] args) throws InterruptedException {
            long now = System.currentTimeMillis();
            CyclicBarrier1 cb = new CyclicBarrier1();
            cb.count();
            Thread.sleep(100);
            long end = System.currentTimeMillis();
            System.out.println(end - now);
        }

    最终输出结果:

     其中1194ms证明了会阻塞主线程。另外,关注Java知音公众号,回复“后端面试”,送你一份面试题宝典!

    https://mp.weixin.qq.com/s/ea1v96Q383eJGQeZsmObFQ

    https://www.cnblogs.com/heiz123/p/15239757.html

    故乡明
  • 相关阅读:
    在人生路上对我影响最大的三位老师
    秋季学期学习总结
    转载非原创 Windows编程革命简史
    转载 关于12360系统的讨论
    SQLServer 触发器
    sqlserver 自定义函数
    jQuery 动画
    jQuery让页面生动起来(操作页面里面的元素)
    jQuery选择元素
    SqlServer_Case_When用法
  • 原文地址:https://www.cnblogs.com/luweiweicode/p/15157855.html
Copyright © 2011-2022 走看看