由来
CountDownLatch虽然好用,但是也有不足,就是它的计数器不能循环使用,是属于一次性的消费品,对此CyclicBarrier倒是弥补了它这个不足,大部分情况下CyclicBarrier是对CountDownLatch的补充,但是两者还是有一点小小的区别
例子
假如公司团建,大家一起做大巴车,在大巴车出发之前,肯定是需要点名的,只有大家都到车上之后,才会发车,然后到了到了目的地之后,肯定是所有人都下车了,司机才能把车开走,这个过程中涉及了2次大家都就位之后,司机才能继续操作,可以证明CyclicBarrier可以循环使用计数器。
class Flag{
public int num=0;
public Flag(){}
public int getNum() {
return num;
}
public void setNum(int num) {
this.num = num;
}
}
public class CyclicBarrierTest {
public static void main(String[] args) {
Flag flag=new Flag();
CyclicBarrier cyclicBarrier = new CyclicBarrier(5, () -> {
if(flag.getNum()==0){
System.out.println("所有人全部上车了,可以发车了");
flag.setNum(1);
}else {
System.out.println("所有人都下车了,司机可以走了");
}
});
for (int i = 1; i < 6; i++) {
new Thread(() -> {
try {
System.out.println(Thread.currentThread().getName() + "号上车了");
cyclicBarrier.await();
System.out.println(Thread.currentThread().getName() + "号开始休息了");
TimeUnit.MILLISECONDS.sleep(new Random().nextInt(2000));
System.out.println(Thread.currentThread().getName() + "号下车了");
cyclicBarrier.await();
System.out.println(Thread.currentThread().getName() + "号到达目的地了");
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
e.printStackTrace();
}
}, String.valueOf(i)).start();
}
System.out.println("主线程不阻塞");
}
}
执行结果
总结
-
CountDownLatch属于一个对多个的(CountDownLatch这篇里有讲到过),而CyclicBarrier是多对多的,只要多个线程都达到后,自会执行接下来的事,没有CountDownLatch的一个等多个,多个等一个的现象。
-
CountDownLatch的计数器只能使用一次,而CyclicBarrier可以循环使用。