透过源码和实例学习CountDownLatch类

最近在做一个程序的 时候遇到了这样的情况,在 主类中需要执行一些操作,同时主类会启动几个线程,在这些线程执行前和执行后都会一系列的操作。因此就用到了CountDownLatch类。CountDownLatch是一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待。

 

下边是一个例子:

import mulithread.CountDownLatch;

class Driver {  
	public static void main(String []args) throws InterruptedException {
		CountDownLatch startSignal = new CountDownLatch(1,"startsignal");
		CountDownLatch doneSignal = new CountDownLatch(2,"endsignal");
		
		for (int i = 0; i < 2; ++i)
			new Thread(new Worker(startSignal, doneSignal,i)).start();// create and start threads
		Thread.sleep(3000);
		System.out.println("don't let run yet"); //don't let run yet
		startSignal.countDown(); //let all threads proceed
		doneSignal.await(); // wait for all to finish
		System.out.println("wait for all to finish");
	}
}

 

import mulithread.CountDownLatch;


class Worker implements Runnable {
	private final CountDownLatch startSignal;
	private final CountDownLatch doneSignal;
	private int threadid;
	Worker(CountDownLatch startSignal, CountDownLatch doneSignal,int threadid) {
		this.startSignal = startSignal;
		this.doneSignal = doneSignal;
		this.threadid=threadid;
	} 

	public void run() {
		try {
			System.out.println(threadid+"before process");
			startSignal.await();//所有调用 await 的线程都一直在入口处等待
			doWork();
			doneSignal.countDown();
			System.out.println(threadid+"after process");
		} catch (InterruptedException ex) {
		
		}  
	}

	void doWork() {
		try {
			Thread.sleep(20);
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
		System.out.println(threadid+"runing----------");
	}
}

 

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.AbstractQueuedSynchronizer;

public class CountDownLatch {
	
    private static final class Sync extends AbstractQueuedSynchronizer {
        private static final long serialVersionUID = 4982264981922014374L;

        Sync(int count) {
            setState(count);
        }

        int getCount() {
            return getState();
        }

        public int tryAcquireShared(int acquires) {
            return getState() == 0? 1 : -1;
        }

        public boolean tryReleaseShared(int releases) {
            // Decrement count; signal when transition to zero
            for (;;) {
                int c = getState();
                if (c == 0)
                    return false;
                int nextc = c-1;
                if (compareAndSetState(c, nextc))
                    return nextc == 0;
            }
        }
    }
    private String name;
    private final Sync sync;

    public CountDownLatch(int count,String name) {
    	this.name=name;
        if (count < 0) throw new IllegalArgumentException("count < 0");
        this.sync = new Sync(count);
    }

    public void await() throws InterruptedException {
        sync.acquireSharedInterruptibly(1);
    	System.out.println(name+" await invoked!"+getCount());
    }
    public boolean await(long timeout, TimeUnit unit)
        throws InterruptedException {
        return sync.tryAcquireSharedNanos(1, unit.toNanos(timeout));
    }

    public void countDown() {
        sync.releaseShared(1);
        System.out.println(name+" count down invoked!"+getCount());
    }
    public long getCount() {
        return sync.getCount();
    }

    public String toString() {
        return super.toString() + "[Count = " + sync.getCount() + "]";
    }
}

 

注意,有三个类,其中Driver是主类,Worker类是要工作时候用到的线程,CountDownLatch类值从java源代码抠出来的,只是做了一些状态输出工作。在这个例子中,有两个工作线程,所有的线程在工作之前(dowork方法完成工作任务)startSignal的await()方法会一直阻塞,直到主类中startSignal的countDown()执行之后才会执行。 同样的,在所有工作完成之前,主类中doneSignal的await()方法会一直阻塞,一直到doneSignal的所有线程的 countdown方法会被调用,使得计数器的值递减,从而后边的代码才会执行。这个例子其实简单而言描述这样一个场景:主类中如果有几个线程,但是这些线程又必须在主类中的一些操作完成之前才可以执行,那么就可以把代码放在主类的startSignal的countdown()方法之上,同样,如果想要在所有线程执行之后,做一些工作,那么就可以把代码放在doneSignal的await()方法之后执行。至于CountDownLatch 中的参数,其实就是计数器的值,看你的具体需求情况啦!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值