写在前面:
CyclicBarrier从字面上来看是指一道屏障,CyclicBarrier的作用是在一组线程到达屏障时,会被阻塞,直到最后一个线程到达屏障时,屏障才会开门,这时,被屏障拦截的线程才会继续运行。
CyclicBarrier构造器
public CyclicBarrier(int parties) {
this(parties, null);
}
public CyclicBarrier(int parties, Runnable barrierAction) {
if (parties <= 0) throw new IllegalArgumentException();
this.parties = parties;
this.count = parties;
this.barrierCommand = barrierAction;
}
CyclicBarrier提供了上述两个构造器,第一个CyclicBarrier构造器中的参数parties表示屏障拦截的线程数量。每个线程调用await方法后,CyclicBarrier会得知线程已经到达了屏障,然后当前线程会被阻塞。直到构造器的parties减为0之后,所有的线程才会被放行。
下面来看一个测试代码:
package cyclicBarrier;
import java.util.concurrent.CyclicBarrier;
public class CyclicBarrierDemo {
static CyclicBarrier c = new CyclicBarrier(2);
public static void main(String[] args) {
new Thread(new Runnable() {
@Override
public void run() {
try {
c.await();
} catch (Exception e) {
e.printStackTrace();
}
System.out.println(1);
}
}).start();
try {
c.await();
} catch (Exception e) {
e.printStackTrace();
}
System.out.println(2);
}
}
由于主线程和子线程的调度是由CPU决定,所以当main函数中调用了2次await方法,也就是说拦截到了足够的线程,就会对所有的线程放行。
如果我们在初始化CyclicBarrier的时候,将参数改成一个大于2的数值,那么程序中因为只有两个线程,拦截不到更多的线程,所以2个线程就永远也不会被执行。
此外,CyclicBarrier提供的第二个构造器,需要传递一个Runnable barrierAction,这样线程到达屏障时,程序会保证barrierAction的优先执行,然后当拦截到线程数量之后,会对其他线程放行。
第二个CyclicBarrier的demo如下:
package cyclicBarrier;
import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier;
public class CyclicBarrierDemo {
static CyclicBarrier c = new CyclicBarrier(2,new A());
public static void main(String[] args) throws Exception {
new Thread(new Runnable() {
@Override
public void run() {
try {
c.await();
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("线程1");
}
}).start();
c.await();
System.out.println("主线程");
}
static class A implements Runnable{
@Override
public void run() {
System.out.println("线程A");
}
}
}
控制台输出:
线程A
线程1
主线程
CyclicBarrier和CountDownLathch的对比
(1)CountDownLatch的计数器只能使用一次。
(2)CyclicBarrier的计数器可以用reset方法重置。
(3)在CyclicBarrier也提供了getNumberWaiting会获得CyclicBarrier阻塞的线程数量。
(4)在CyclicBarrier也提供了isBroken方法来了解阻塞的线程是否被中断。