CyclicBarrier的用法
来源:互联网 发布:哪个洗衣店好 知乎 编辑:程序博客网 时间:2024/05/16 11:31
CyclicBarrier是一个同步辅助类,它允许一组线程互相等待,直到到达某个公共屏障点 (common barrier point)。在涉及一组固定大小的线程的程序中,这些线程必须不时地互相等待,此时 CyclicBarrier 很有用。因为该 barrier 在释放等待线程后可以重用,所以称它为循环 的 barrier。
CyclicBarrier类似于CountDownLatch也是个计数器, 不同的是CyclicBarrier数的是调用了CyclicBarrier.await()进入等待的线程数, 当线程数达到了CyclicBarrier初始时规定的数目时,所有进入等待状态的线程被唤醒并继续。 CyclicBarrier就象它名字的意思一样,可看成是个障碍, 所有的线程必须到齐后才能一起通过这个障碍。 CyclicBarrier初始时还可带一个Runnable的参数,此Runnable任务在CyclicBarrier的数目达到后,所有其它线程被唤醒前被执行。
CyclicBarrier(int parties)
创建一个新的 CyclicBarrier,它将在给定数量的参与者(线程)处于等待状态时启动,但它不会在每个 barrier 上执行预定义的操作。
CyclicBarrier(int parties, Runnable barrierAction)
创建一个新的 CyclicBarrier,它将在给定数量的参与者(线程)处于等待状态时启动,并在启动 barrier 时执行给定的屏障操作,该操作由最后一个进入 barrier 的线程执行
int
await()
在所有
参与者
都已经在此 barrier 上调用 await 方法之前,将一直等待。int
await(long timeout, TimeUnit unit)
在所有
参与者
都已经在此屏障上调用 await 方法之前,将一直等待。int
getNumberWaiting()
返回当前在屏障处等待的参与者数目。
int
getParties()
返回要求启动此 barrier 的参与者数目。
boolean
isBroken()
查询此屏障是否处于损坏状态。
void
reset()
将屏障重置为其初始状态。
package com.thread;import java.util.concurrent.CyclicBarrier;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;import java.util.concurrent.Semaphore;public class CyclicBarrierTest { public static void main(String[] args) { ExecutorService service = Executors.newCachedThreadPool(); final CyclicBarrier cb = new CyclicBarrier(3);//创建CyclicBarrier对象并设置3个公共屏障点 for(int i=0;i<3;i++){ Runnable runnable = new Runnable(){ public void run(){ try { Thread.sleep((long)(Math.random()*10000)); System.out.println("线程" + Thread.currentThread().getName() + "即将到达集合地点1,当前已有" + cb.getNumberWaiting() + "个已经到达,正在等候"); cb.await();//到此如果没有达到公共屏障点,则该线程处于等待状态,如果达到公共屏障点则所有处于等待的线程都继续往下运行 Thread.sleep((long)(Math.random()*10000)); System.out.println("线程" + Thread.currentThread().getName() + "即将到达集合地点2,当前已有" + cb.getNumberWaiting() + "个已经到达,正在等候"); cb.await(); Thread.sleep((long)(Math.random()*10000)); System.out.println("线程" + Thread.currentThread().getName() + "即将到达集合地点3,当前已有" + cb.getNumberWaiting() + "个已经到达,正在等候"); cb.await(); } catch (Exception e) { e.printStackTrace(); } } }; service.execute(runnable); } service.shutdown(); }}
线程pool-1-thread-1即将到达集合地点1,当前已有0个已经到达,正在等候线程pool-1-thread-3即将到达集合地点1,当前已有1个已经到达,正在等候线程pool-1-thread-2即将到达集合地点1,当前已有2个已经到达,正在等候线程pool-1-thread-3即将到达集合地点2,当前已有0个已经到达,正在等候线程pool-1-thread-2即将到达集合地点2,当前已有1个已经到达,正在等候线程pool-1-thread-1即将到达集合地点2,当前已有2个已经到达,正在等候线程pool-1-thread-1即将到达集合地点3,当前已有0个已经到达,正在等候线程pool-1-thread-3即将到达集合地点3,当前已有1个已经到达,正在等候线程pool-1-thread-2即将到达集合地点3,当前已有2个已经到达,正在等候
如果在构造CyclicBarrier对象的时候传了一个Runnable对象进去,则每次到达公共屏障点的时候都最先执行这个传进去的Runnable,然后再执行处于等待的Runnable。如果把上面的例子改成下面这样:
package com.thread;import java.util.concurrent.CyclicBarrier;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;import java.util.concurrent.Semaphore;public class CyclicBarrierTest { public static void main(String[] args) { ExecutorService service = Executors.newCachedThreadPool(); //final CyclicBarrier cb = new CyclicBarrier(3);//创建CyclicBarrier对象并设置3个公共屏障点 final CyclicBarrier cb = new CyclicBarrier(3,new Runnable(){ @Override public void run() { System.out.println("********我最先执行***********"); } }); for(int i=0;i<3;i++){ Runnable runnable = new Runnable(){ public void run(){ try { Thread.sleep((long)(Math.random()*10000)); System.out.println("线程" + Thread.currentThread().getName() + "即将到达集合地点1,当前已有" + cb.getNumberWaiting() + "个已经到达,正在等候"); cb.await();//到此如果没有达到公共屏障点,则该线程处于等待状态,如果达到公共屏障点则所有处于等待的线程都继续往下运行 Thread.sleep((long)(Math.random()*10000)); System.out.println("线程" + Thread.currentThread().getName() + "即将到达集合地点2,当前已有" + cb.getNumberWaiting() + "个已经到达,正在等候"); cb.await(); //这里CyclicBarrier对象又可以重用 Thread.sleep((long)(Math.random()*10000)); System.out.println("线程" + Thread.currentThread().getName() + "即将到达集合地点3,当前已有" + cb.getNumberWaiting() + "个已经到达,正在等候"); cb.await(); } catch (Exception e) { e.printStackTrace(); } } }; service.execute(runnable); } service.shutdown(); }}
则结果如下:
线程pool-1-thread-1即将到达集合地点1,当前已有0个已经到达,正在等候线程pool-1-thread-3即将到达集合地点1,当前已有1个已经到达,正在等候线程pool-1-thread-2即将到达集合地点1,当前已有2个已经到达,正在等候********我最先执行***********线程pool-1-thread-1即将到达集合地点2,当前已有0个已经到达,正在等候线程pool-1-thread-3即将到达集合地点2,当前已有1个已经到达,正在等候线程pool-1-thread-2即将到达集合地点2,当前已有2个已经到达,正在等候********我最先执行***********线程pool-1-thread-1即将到达集合地点3,当前已有0个已经到达,正在等候线程pool-1-thread-3即将到达集合地点3,当前已有1个已经到达,正在等候线程pool-1-thread-2即将到达集合地点3,当前已有2个已经到达,正在等候********我最先执行***********
阅读全文
0 0
- CyclicBarrier的用法
- 多线程CyclicBarrier的用法
- CyclicBarrier的用法
- CyclicBarrier的用法
- CyclicBarrier的用法
- JUC--CyclicBarrier的用法
- CyclicBarrier的用法
- CyclicBarrier的用法
- CyclicBarrier的用法
- Java CyclicBarrier的用法
- CountDownLatch和CyclicBarrier 的用法
- 栅栏(CyclicBarrier类)的用法
- Java中CyclicBarrier的用法
- CyclicBarrier用法
- CyclicBarrier用法
- CyclicBarrier用法
- Java多线程 - 浅析CyclicBarrier的用法
- java中CyclicBarrier的简单用法
- Vue组件中的slot
- Spring配置中对静态资源的正确引用!
- linux 链接命令
- 用servlet记录用户访问次数
- spring框架依赖注入与控制反转
- CyclicBarrier的用法
- 蓝桥杯第六届决赛:四阶幻方
- HDU 1171 Big Event in HDU (单调队列优化多重背包)
- 算法中的NP问题
- ubuntu14.04 openssh-server安装问题
- 对常见的通信方式协议的理解
- PHP数组常用函数总结
- 实例方法与实例事件
- 股票数据API整理