多线程——CyclicBarrier的使用

CyclicBarrier的字面意思是可循环使用(Cyclic)的屏障(Barrier)。它要做的事情是,让一 组线程到达一个屏障(也可以叫同步点)时被阻塞,直到最后一个线程到达屏障时,屏障才会开门,所有被屏障拦截的线程才会继续运行。
CyclicBarrier默认的构造方法是CyclicBarrier(int parties),其参数表示屏障拦截的线程数量,每个线程调用await方法告诉CyclicBarrier我已经到达了屏障,然后当前线程被阻塞。
代码示例:

public static void main(String[] args) {
        CyclicBarrier barrier=new CyclicBarrier(2);
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    barrier.await();
                } catch (Exception e) {
                    e.printStackTrace();
                }
                System.out.println(1);
            }
        }).start();

        try {
            barrier.await();
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println(2);

    }

运行结果:

2
1

Process finished with exit code 0

如果把new CyclicBarrier(2)修改成new CyclicBarrier(3),则主线程和子线程会永远等待, 因为没有第三个线程执行await方法,即没有第三个线程到达屏障,所以之前到达屏障的两个 线程都不会继续执行。

CyclicBarrier初始化是还可以带一个Runnable的参数,这个Runnable任务在 CyclicBarrier的数目到达后,所有线程被唤醒前执行。

public static void main(String[] args) {
        CyclicBarrier barrier=new CyclicBarrier(5, new Runnable() {
            @Override
            public void run() {
                System.out.println("所有线程开始执行");
            }
        });

        for(int i=0;i<5;i++){
            final int k=i;
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                        barrier.await();
                        System.out.println(k);
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            }).start();
        }

    }
所有线程开始执行
4
3
2
1
0

Process finished with exit code 0

可以看到 CyclicBarrier参数里的Runnable任务是第一个执行的。但是假如将 barrier.await();放在System.out.println(k);后面的话,“所有线程开始执行”这句话就会是最后一个输出的。

CyclicBarrier可以用于多线程计算数据,最后合并计算结果的场景。

代码:

 public static void main(String[] args) {
        ConcurrentHashMap<String,Integer> map=new ConcurrentHashMap<>();
        CyclicBarrier barrier=new CyclicBarrier(5, new Runnable() {
            @Override
            public void run() {
                int count=0;
                for(Map.Entry<String,Integer> e:map.entrySet()){
                    count+=e.getValue();
                }
                System.out.println(count);
            }
        });

        for(int i=0;i<5;i++){
            final int k=i;
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                       map.put(Thread.currentThread().getName(),k);
                        barrier.await();
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            }).start();
        }

    }

最后我们就可以计算每一个map值中的总量
运行结果:

10

Process finished with exit code 0

CountDownLatch和CyclicBarrier的区别
CountDownLatch的计数器只能使用一次,而CyclicBarrier的计数器可以使用reset()方法重 置。所以CyclicBarrier能处理更为复杂的业务场景。例如,如果计算发生错误,可以重置计数 器,并让线程重新执行一次。 CyclicBarrier还提供其他有用的方法,比如getNumberWaiting方法可以获得Cyclic-Barrier 阻塞的线程数量。isBroken()方法用来了解阻塞的线程是否被中断。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值