[关闭]
@boothsun 2017-07-31T05:27:22.000000Z 字数 4324 阅读 1118

CyclicBarrier的使用

Java多线程


简介

CyclicBarrier的字面意思是可循环使用(Cyclic)的屏障(Barrier)。CyclicBarrier的作用是让一组线程之间相互等待,任何一个线程到达屏障点后就阻塞,直到最后一个线程到达,才都继续往下执行。个人理解:CyclicBarrier可以看成是一道大门或者关卡,先到的线程会被阻塞在大门口,直到最后一个线程到达屏障时,大门才被打开,所有被阻塞的线程才会继续干活。就像是朋友聚餐,只有最后一个朋友到达时,才会开吃!

屏障(Barrier)

CyclicBarrier 的构造函数可以传入一个整数,其含义是屏障可拦截的线程数,每个线程都可调用await方法告诉CyclicBarrier“我已经到达了屏障前”,CyclicBarrier内部是执行+1操作,一旦屏障前阻塞的线程数等于构造函数传入的可拦截线程数时,所有被阻塞的线程都将被唤醒,继续往下执行。

常用API

  1. //parties表示屏障前可阻塞的线程数,当阻塞的线程数到达parties时,屏障被打开,所有阻塞的线程将会被唤醒
  2. public CyclicBarrier(int parties);
  3. // 此构造方法不同于上面的是在屏障被打开时将优先执行barrierAction,方便处理更负责的业务场景
  4. public CyclicBarrier(int parties, Runnable barrierAction) ;
  5. // 等待屏障的打开
  6. public int await() throws InterruptedException,BrokenBarrierException ;
  7. //等待屏障的打开 超时会抛出 TimeoutException
  8. public int await(long timeout, TimeUnit unit) throws
  9. InterruptedException,
  10. BrokenBarrierException,
  11. TimeoutException ;
  12. // 返回在屏障前等待的线程数
  13. public int getNumberWaiting() ;
  14. // 获取 当前屏障初始化时 可阻塞的线程数
  15. public int getParties()
  16. /**
  17. * 作用:
  18. * 1. 查询此屏障是否处于损坏状态。
  19. * 产生损坏状态的原因:
  20. * 1. 由于超时或者屏障重置(reset)
  21. * 2. 某个屏障操作抛出异常
  22. */
  23. public boolean isBroken() ;
  24. <div class="md-section-divider"></div>

使用举例

模拟 聚餐场景

  1. public class CyclicBarrierTest {
  2. public static void main(String[] args) throws Exception{
  3. ExecutorService service = Executors.newFixedThreadPool(3) ;
  4. CyclicBarrier cyclicBarrier = new CyclicBarrier(3,()->{
  5. System.out.println("全都到了 【开吃!】");
  6. }) ;
  7. for(int i = 0 ; i < 3 ; i++) {
  8. final int number = i ;
  9. service.execute(()->{
  10. try {
  11. System.out.println("编号:" + number + "开始出发 【去聚餐】");
  12. Thread.sleep((int)(Math.random() * 10000));
  13. System.out.println("编号:" + number + " 【到达聚餐地点】");
  14. cyclicBarrier.await();
  15. } catch (Exception e) {
  16. e.printStackTrace();
  17. }
  18. });
  19. }
  20. service.shutdown();
  21. }
  22. }
  23. <div class="md-section-divider"></div>

循环使用

循环使用指的是在大门被打开后,可以再次关闭;即再让之前指定数目的线程在屏障前阻塞等待,然后再次打开大门。

方法reset()的作用就是重置屏障,以保证循环使用。

常用API

  1. // 将屏障重置为其初始化状态即重置为构造函数传入的parties值。
  2. public void reset()
  3. <div class="md-section-divider"></div>

使用举例

  1. import java.util.concurrent.CyclicBarrier;
  2. import java.util.concurrent.Executors;
  3. import java.util.concurrent.ThreadPoolExecutor;
  4. /**
  5. * Created by 58 on 2017-7-14.
  6. */
  7. public class CyclicBarrierTest {
  8. public static void main(String[] args) throws Exception{
  9. ThreadPoolExecutor service = (ThreadPoolExecutor)Executors.newFixedThreadPool(3) ;
  10. CyclicBarrier cyclicBarrier = new CyclicBarrier(3,()->{
  11. System.out.println("全都到了 【开吃!】");
  12. }) ;
  13. for(int i = 0 ; i < 3 ; i++) {
  14. final int number = i ;
  15. service.execute(()->{
  16. try {
  17. System.out.println("编号:" + number + "开始出发 【去聚餐】");
  18. Thread.sleep((number+2) * 1000);
  19. System.out.println("编号:" + number + " 【到达聚餐地点】");
  20. cyclicBarrier.await();
  21. } catch (Exception e) {
  22. e.printStackTrace();
  23. }
  24. });
  25. }
  26. Thread.sleep( 3 * 1000);
  27. System.out.println(cyclicBarrier.getNumberWaiting());
  28. cyclicBarrier.reset();
  29. System.out.println(cyclicBarrier.getNumberWaiting());
  30. }
  31. }

运行截图

说明

关于reset方法的说明 :

/**
* Resets the barrier to its initial state. If any parties are
* currently waiting at the barrier, they will return with a
* {@link BrokenBarrierException}. Note that resets after
* a breakage has occurred for other reasons can be complicated to
* carry out; threads need to re-synchronize in some other way,
* and choose one to perform the reset. It may be preferable to
* instead create a new barrier for subsequent use.
*/

上面是JDK中reset方法的注释,简单翻译成中文如下:

将屏障重置为其初始状态。此时,如果有任何的一个参与者正在屏障前等待,它将会返回一个 BrokenBarrierException异常。注意:如果因为其他原因使屏障发生损坏,此时屏障的重置将会变得很复杂;为了将来的使用,相比需要考虑使用其他方式重新同步线程,并选择其中一个线程来执行重置,更好的解决办法是创建一个新的屏障。

从上面的翻译,我们可以得出结论:

关于BrokenBarrierException异常的说明:

/**
* Exception thrown when a thread tries to wait upon a barrier that is
* in a broken state, or which enters the broken state while the thread
* is waiting.
*/

上面是JDK中BrokenBarrierException方法的注释,简单翻译成中文如下:

当某个线程试图在一个已经处于损坏状态的屏障前等待或者在线程等待时屏障进入了损坏状态 将会抛出此异常。

与CountDownLatch的比较

一对多与多对多

CountDownLatch一般用于某个线程A等待若干其他线程执行完任务之后,它才执行。而CyclicBarrier一般用于多个线程之间相互等待,比如需要在一个同时执行时间点上达成一致,然后同时开启一项工作;重点是“多个线程之间” 任何一个线程没有完成任务,则其他所有的线程都必须等待。

循环使用

CyclicBarrier的reset方法可以重置屏障。

CyclicBarrier的应用场景

CyclicBarrier可以用于多线程计算数据,最后合并计算结果的应用场景。比如我们用一个Excel保存了用户所有银行流水,每个sheet保存一个账户近一年的每笔银行流水,现在需要统计用户的日均银行流水,先用多线程处理每个sheet里的银行流水,都执行完之后,得到每个sheet的日均银行流水,最后,再用barrierAction来汇总这些线程的计算结果,计算出整个Excel的日均银行流水。

barrierAction的执行时机

下面的注释片段是JDK源码中,关于CyclicBarrier的介绍。

*A {@code CyclicBarrier} supports an optional {@link Runnable} command
* that is run once per barrier point, after the last thread in the party
* arrives, but before any threads are released.
* This barrier action is useful
* for updating shared-state before any of the parties continue.
*

after the last thread in the party arrives , but before any threads are released 这句明确说明barrierAction的执行时机:在最后一个线程到达后,但是在所有线程被释放运行前。

线程回收问题

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注