简单使用CyclicBarrier(栏栅)

1.需求分析
使用CyclicBarrier类模拟赛跑功能:新建N个人选手,当所有选手准备好时,开始前进一次,每次前进的距离随机在1-3之间,所有选手前进一次后,裁判开始检查有没有到达终点的选手,没有则继续之前操作,直到有一人或多人同时到达终点。
2.Person类
public class Person extends Thread{
	private final String name;
	private CyclicBarrier cb;
	//本次前进之后的总距离
	private int now;
	
	public Person(String name,CyclicBarrier cb){
		this.name = name;
		this.cb = cb;
	}
	public synchronized int getNow(){
		return now;
	}
	public synchronized String getPersonName(){
		return name;
	}
	
	@Override
	public void run(){
		while(true){
			try {
				cb.await();
				int random = (int)(Math.random()*3+1);
				synchronized (this) {
					now += random;
					System.out.println(name +"本次跑了 "+random+" 米,共跑了 "+now+" 米");
				}
			} catch (InterruptedException e) {
				e.printStackTrace();
			} catch (BrokenBarrierException e) {
				e.printStackTrace();
			}
		}
			
	} 

}
3.Judge类
public class Judge extends Thread{
	private static final int MAX = 100;
	private CopyOnWriteArrayList<Person> cowl; 
	
	public synchronized void setCowl(CopyOnWriteArrayList<Person> cowl){
		this.cowl = cowl;
	}
	private void judge(){
	    System.out.println("\n==============start judge=================");
		/*
		try {
			Thread.sleep(1000);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}*/
		List<Person> list  = new ArrayList<>();
		for(Person p :cowl){
			if(p.getNow()>=MAX)
				list.add(p);
		}
		if(list.size()!=0){
			for(Person p:list)
				System.out.println(p.getPersonName()+"率先抵达终点!");
			System.exit(0);
		}
		System.out.println("\n==============end judge=================");
	}
	@Override
	public void run() {
		judge();
	}
}
4.测试启动类
public class CyclicBarrierTest {
	
	public static void main(String[] args) {
	
		Judge judge = new Judge();
		CyclicBarrier cb = new CyclicBarrier(10, judge);
		
		List<Person> list = new ArrayList<>();
		for(int i=0;i<10;i++){
			Person p = new Person("tom"+i,cb);
			p.start();
			list.add(p);
		}
		judge.setCowl(new CopyOnWriteArrayList<>(list));
	}
}
5.运行结果(节选)
(start)
==============start judge=================
==============end judge=================

tom0本次跑了 2 米,共跑了 2 米
tom3本次跑了 1 米,共跑了 1 米
tom1本次跑了 1 米,共跑了 1 米
tom5本次跑了 1 米,共跑了 1 米
tom9本次跑了 3 米,共跑了 3 米
tom6本次跑了 3 米,共跑了 3 米
tom8本次跑了 2 米,共跑了 2 米
tom2本次跑了 3 米,共跑了 3 米
tom4本次跑了 2 米,共跑了 2 米
tom7本次跑了 2 米,共跑了 2 米

==============start judge=================
==============end judge=================
...

tom5本次跑了 2 米,共跑了 98 米
tom1本次跑了 3 米,共跑了 97 米
tom8本次跑了 3 米,共跑了 97 米
tom4本次跑了 2 米,共跑了 95 米
tom6本次跑了 2 米,共跑了 91 米
tom9本次跑了 1 米,共跑了 95 米
tom7本次跑了 2 米,共跑了 93 米
tom2本次跑了 2 米,共跑了 92 米
tom3本次跑了 2 米,共跑了 87 米
tom0本次跑了 3 米,共跑了 97 米

==============start judge=================
==============end judge=================

tom0本次跑了 3 米,共跑了 100 米
tom5本次跑了 1 米,共跑了 99 米
tom1本次跑了 2 米,共跑了 99 米
tom8本次跑了 2 米,共跑了 99 米
tom4本次跑了 2 米,共跑了 97 米
tom6本次跑了 1 米,共跑了 92 米
tom9本次跑了 3 米,共跑了 98 米
tom7本次跑了 2 米,共跑了 95 米
tom2本次跑了 2 米,共跑了 94 米
tom3本次跑了 1 米,共跑了 88 米

==============start judge=================
tom0率先抵达终点!
从结果上来看,在一次"循环"中,屏障操作总是在所有参与者线程全部就绪之后启动,屏障操作运行完成之后,参与者线程才开始执行,这一点在judge()方法中的注释打开后可以很清楚的观察到。
于是去阅读源码,果然如此:
 int index = --count;
    if (index == 0) {  // tripped
        boolean ranAction = false;
        try {
            final Runnable command = barrierCommand;
            if (command != null)
                command.run();//此处调用的run();
            ranAction = true;
            nextGeneration();
            return 0;
        } finally {
            if (!ranAction)
                breakBarrier();
        }
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值