1.简述
CountDownLatch是Java1.5之后引入的Java并发工具类(闭锁的一个实现),放在java.util.concurrent包下。用给定的计数初始化CountDownLatch。由于调用了countDown方法,所以在当前计数到达零之前,await方法会一直受阻塞。之后,会释放所有等待的线程,await的所有后续调用都将立即返回。这种现象只出现一次——计数无法被重置。如果需要重置计数,请考虑使用CyclicBarrier。
CountDownLatch能够使一个或多个线程等待其他线程完成各自的工作后再执行。
闭锁(Latch):一种同步方法,可以延迟线程的进度直到线程到达某个终点状态。
其他的N个线程必须引用闭锁对象,因为他们需要通知CountDownLatch对象,他们已经完成了各自的任务,这种机制就是通过countDown()方法来完成的。
CountDownLatch优点:
- 对使用者而言,你只需要传入一个int型变量控制任务数量即可,至于同步队列的出队入队维护,state变量值的维护对使用者都是透明的,使用方便。
CountDownLatch缺点:
- CountDownLatch设置了state后就不能更改,也不能循环使用。
CountDownLatch使用场景:
- 确保某个计算在其需要的所有资源都被初始化之后才继续执行。
- 确保某个服务在其依赖的所有其他服务都已启动后才启动。
- 等待知道某个操作的所有者都就绪在继续执行。
2.CountDownLatch常用方法
/**构造方法 */ //构建了一个 CountDownLatch与给定的计数初始化。 CountDownLatch(int count) /**常用方法 */ //获取当前count的值。 long getCount() //让当前线程在此CountDownLatch对象上等待,可以中断。与notify()、notifyAll()方法对应。(注意:wait()方法是从Object类继承来的) void wait() //让当前线程等待此CountDownLatch对象的count变为0,可以中断。 void await() //让当前线程等待此CountDownLatch对象的count变为0,可以超时、可以中断。 boolean await(long timeout, TimeUnit unit) //使此CountDownLatch对象的count值减1(无论执行多少次,count最小值为0)。 void countDown()
3.CountDownLatch的源码分析
CountDownLatch内部实现是依赖于AQS共享锁(共享模式)来实现的。下面,我们分析CountDownLatch中的源码。
CountDownLatch的主要属性:
//Sync内部类,实现AQS的state private static final class Sync extends AbstractQueuedSynchronizer { private static final long serialVersionUID = 4982264981922014374L; // 传入初始次数 Sync(int count) { setState(count); } // 获取还剩的次数 int getCount() { return getState(); } //尝试获取共享锁 protected int tryAcquireShared(int acquires) { //注意:这里state等于0的时候返回的是1,也就是说count减为0的时候获取总是成功 //state不等于0的时候返回的是-1,也就是count不为0的时候总是要排队 return (getState() == 0) ? 1 : -1; } //尝试释放锁 protected boolean tryReleaseShared(int releases) { for (;;) { //state的值 int c = getState(); //等于0了,则无法再释放了 if (c == 0) return false; //将count的值减1 int nextc = c-1; //原子更新state的值 if (compareAndSetState(c, nextc)) //减为0的时候返回true,这时会唤醒后面排队的线程 return nextc == 0; } } } //AQS同步器的实现类 private final Sync sync;
CountDownLatch的构造函数:
/**需要传入一个count变量,也就是需要等待的线程数。 */ public CountDownLatch(int count) { if (count < 0) throw new IllegalArgumentException("count < 0"); //初始化状态数 this.sync = new Sync(count); }
CountDownLatch的await方法:
/**该方法被调用时将会使当前线程在锁存器倒计数至零之前一直等待,除非线程被中断。 */ public void await() throws InterruptedException { // 调用AQS的acquireSharedInterruptibly方法 sync.acquireSharedInterruptibly(1); } /**该方法被调用时将会使当前线程在锁存器倒计数至零之前一直等待,除非线程被中断或超出了指定的等待时间。 */ public boolean await(long timeout, TimeUnit unit) throws InterruptedException { // 调用AQS的tryAcquireSharedNanos方法 return sync.tryAcquireSharedNanos(1, unit.toNanos(timeout)); }
CountDownLatch的countDown方法:
public void countDown() { // 调用AQS的释放共享锁方法 sync.releaseShared(1); }
4.CountDownLatch使用示例
public class Test { public static void main(String[] args) throws Exception { /*创建CountDownLatch实例,计数器的值初始化为5*/ final CountDownLatch downLatch = new CountDownLatch(5); /*创建三个线程,每个线程等待2s,表示执行比较耗时的任务*/ for(int i = 0;i < 5;i++){ new Thread(new Runnable() { public void run() { try { Thread.sleep(2000); System.out.println(String.format("线程%s已完成", Thread.currentThread().getName())); /*任务完成后调用CountDownLatch的countDown()方法,进行减1*/ downLatch.countDown(); }catch (InterruptedException e){ e.printStackTrace(); } } }).start(); } /*主线程调用await()方法,等到其他五个线程执行完后才继续执行*/ downLatch.await(); System.out.print("所有线程都已经执行完成,继续运行主线程逻辑"); } }
5.总结
经过分析CountDownLatch的源码可知,其底层结构仍然是AQS,对其线程所封装的结点是采用共享模式,而ReentrantLock是采用独占模式。
CountDownLatch主要是通过计数器state来控制是否可以执行其他操作,如果不能就通过LockSupport.park方法挂起线程,直到其他线程执行完毕后唤醒它。