Java并发编程之CyclicBarrier使用指南

一、CyclicBarrier能做什么事情

和CountDownLatch一样,CyclicBarrier也是java.util.concurrent包下的一个类;从类名就可以看出,这是一个可以循环使用(Cylcic)的屏障(Barrier),所做的事情就是让一组线程到达一个屏障(同步点)时被阻塞,直到这组线程中的最后一个到达屏障时,屏障才会打开,之前阻塞的线程继续运行。过程如下图所示

CyclicBarrier运行图

上图中的三个线程中各有一个barrier.await,任何一个线程在运行到barrier.await时都会进入阻塞等待状态,直到三个线程都到了barrier.await时才从await返回,继续向后运行。

二、CyclicBarrier 如何使用

实例化CyclicBarrier对象时通过它的构造函数设置屏障要拦截的线程(调用barrier.await的次数)的数据量,每个线程通过调用CyclicBarrier实例的await方法告诉CyclicBarrier我已经到达屏障,并将自己阻塞。

此外,如果在构造CyclicBarrier时设置了一个Runnable实现,那么最后一个barrier.await 的线程会执行这个方法,以完成一些预设工作

CyclicBarrier经常用于多线程计算数据,最后要将计算结果合并的场景。例如一个Excel表记录了用户一个季度所有的银行流水,每个sheet记录了该用户每个月的银行流水情况,要统计该用户整个季度的银行流水状况时,可以先使用多线程统计每个sheet的银行流水,都执行完毕后,使用每个线程的计算结果来计算出该用户整个季度的银行流水状况。

public class CyclicBarrierTest implements Runnable{

    /*创建一个CyclicBarrier实例,屏障数据设为3,处理完之后执行当前类的run方法*/
    private CyclicBarrier cb = new CyclicBarrier(3,this);

    /*创建线程池,只有三个月的数据,所以只需三个线程*/
    private Executor executor = Executors.newFixedThreadPool(3);

    /*创建一个ConcurrentHashMap,用来保存每个sheet计算出的结果*/
    private ConcurrentHashMap<String,Integer> sheetBankWaterCount = new ConcurrentHashMap<String, Integer>();

    public void count() {
        for(int i = 0;i<3;i++){

            /*每个线程用来处理单个sheet中的任务*/
            executor.execute(new Runnable() {

                public void run() {

                    /*此处加入复杂的逻辑处理代码*/
                    sheetBankWaterCount.put(Thread.currentThread().getName(),1);

                    try {

                        /*线程完成工作后调用await 设置屏障*/
                        cb.await();
                    }catch (BrokenBarrierException e){
                        e.printStackTrace();
                    }catch (InterruptedException e){
                        e.printStackTrace();
                    }

                }
            });
        }
    }


    /*等到所有的*/

    public void run() {
        int res = 0;
        /*根据之前多线程的结果计算出整个季度的银行流水*/
        for (Map.Entry<String,Integer> sheet: sheetBankWaterCount.entrySet()) {
            res += sheet.getValue();
        }

        sheetBankWaterCount.put("result",res);
        /*将结果输出*/
        System.out.println(res);
    }

    public static void main(String[] args){

        CyclicBarrierTest test = new CyclicBarrierTest();

        /*注意,此时不需要调用test.run(),最后一个await方法会调用run方法*/
        test.count();
    }
}

三、使用CyclicBarrier时要注意的问题

在线程池中使用CyclicBarrier时一定要注意线程的数量要多于CyclicBarrier实例中设置的阻塞线程的数量就会发生死锁。
调用await()方法的次数一定要等于屏障中设置的阻塞线程的数量,否则也会死锁。

四、CyclicBarrier和CountDownLatch的区别

  1. 首先二者都能让一个或多个线程阻塞等待,都可以用在多个线程间的协调,起到线程同步的作用。但CountDownLatch是多个线程都进行了countDown之后才会触发时间,唤醒await在latch上的线程,执行完countDown操作之后会继续自己线程的工作。而CyclicBarrier是一个栅栏,用于同步所有调用await方法的线程,等到所有的方法都执行了await方法后,所有的线程才会返回各自执行自己的工作。
  2. CountDownLatch计数器只能使用一次,而CyclicBarrier的计数器可以调用 reset() 方法重置,能处理更加复杂的业务场景。

上一篇-java并发之CountDownLatch使用方法

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,386评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,939评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,851评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,953评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,971评论 5 369
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,784评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,126评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,765评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,148评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,744评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,858评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,479评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,080评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,053评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,278评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,245评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,590评论 2 343

推荐阅读更多精彩内容

  • 莫在该拼搏的时候选择安逸,莫在后悔中度过一生。
    1b811b963cdd阅读 181评论 0 0
  • 又是好久没写了!好累的,有时候也不知道写什么,但是我觉得我们是闹别扭了吗?上周六,你和朋友出去玩,什么也没和我说,...
    LiuWaiting阅读 164评论 0 0
  • 我的宝贝盛开是个爱笑的孩子。我爸说,这个孩子整天一副甜不梭的脸,这让我想起朋友也是这么说我的。有人说,爱笑的孩子运...
    毛小生阅读 223评论 3 1
  • 每次夜班之前都有夜班综合症
    小诸诸阅读 127评论 0 0