Java并发编程基础---(7)CyclicBarrier同步屏障总结

写在前面:

    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方法来了解阻塞的线程是否被中断。






评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值