CountDownLatch 是什么
CountDownLatch 是在 java1.5 被引入的,跟它一起被引入的并发工具类还有 CyclicBarrier、Semaphore、ConcurrentHashMap 和 BlockingQueue,它们都存在于 java.util.concurrent 包下。CountDownLatch 这个类能够使一个线程等待其他线程完成各自的工作后再执行。例如,应用程序的主线程希望在负责启动框架服务的线程已经启动所有的框架服务之后再执行。
CountDownLatch 是通过一个计数器来实现的,计数器的初始值为线程的数量。每当一个线程完成了自己的任务后,计数器的值就会减 1。当计数器值到达 0 时,它表示所有的线程已经完成了任务,然后在闭锁上等待的线程就可以恢复执行任务。
CountDownLatch 如何工作
CountDownLatch.java 类中定义的构造函数:
//Constructs a CountDownLatch initialized with the given count.
public void CountDownLatch(int count) {...}
构造器中的计数值(count)实际上就是闭锁需要等待的线程数量。这个值只能被设置一次,而且 CountDownLatch 没有提供任何机制去重新设置这个计数值。
与 CountDownLatch 的第一次交互是主线程等待其他线程。主线程必须在启动其他线程后立即调用**CountDownLatch.await()**方法。这样主线程的操作就会在这个方法上阻塞,直到其他线程完成各自的任务。
其他 N 个线程必须引用闭锁对象,因为他们需要通知 CountDownLatch 对象,他们已经完成了各自的任务。这种通知机制是通过**CountDownLatch.countDown()**方法来完成的;每调用一次这个方法,在构造函数中初始化的 count 值就减 1。所以当 N 个线程都调用了这个方法,count 的值等于 0,然后主线程就能通过 await()方法,恢复执行自己的任务。
在实时系统中的使用场景
1.实现最大的并行性:有时我们想同时启动多个线程,实现最大程度的并行性。例如,我们想测试一个单例类。如果我们创建一个初始计数为 1 的 CountDownLatch,并让所有线程都在这个锁上等待,那么我们可以很轻松地完成测试。我们只需调用 一次 countDown()方法就可以让所有的等待线程同时恢复执行。
2.开始执行前等待 n 个线程完成各自任务:例如应用程序启动类要确保在处理用户请求前,所有 N 个外部系统已经启动和运行了。
3.死锁检测:一个非常方便的使用场景是,你可以使用 n 个线程访问共享资源,在每次测试阶段的线程数目是不同的,并尝试产生死锁。
常见面试题
可以为你的下次面试准备以下一些 CountDownLatch 相关的问题:
解释一下 CountDownLatch 概念?
CountDownLatch 和 CyclicBarrier 的不同之处?
给出一些 CountDownLatch 使用的例子?
CountDownLatch 类中主要的方法?
CountDownLatch(int)
await()
await(long,TimeUnit)
CountDown()
getCount()
toString
在网上看到很多人对于 CountDownLatch 和 CyclicBarrier 的区别简单理解为 CountDownLatch 是一次性的,而 CyclicBarrier 在调用 reset 之后还可以继续使用。
那如果只是这么简单的话,我觉得 CyclicBarrier 简单命名为 ResetableCountDownLatch 好了,显然不是的。
我的理解是,要从他们的设计目的去看这两个类。javadoc 里面的描述是这样的。
CountDownLatch: A synchronization aid that allows one or more threads to wait until a set of operations being performed in other threads completes.
CyclicBarrier : A synchronization aid that allows a set of threads to all wait for each other to reach a common barrier point.
CountDownLatch 和 CyclicBarrier 本质是一样的,都是在每个 Thread 设置 CountDownLatch 或 CyclicBarrier 的屏障点 point, 每个 Thread 计入的 Condition 中队列, 执行到屏障点 point 时候,回调 CountDownLatch 或 CyclicBarrier 的 await 方法,在 Condition 中队列遍历是否所有的 Thread 都已经执行到屏障点 point。但是并没有释放 Thread ,Thread 一致是运行状态(可以理解成时等待其他 Thread 全部完成屏障点 point)。
区别是:
每个 Thread 到达 CountDownLatch 的屏障点 point 可以调用 countDown()计数-1,到全部为计数 0 的时候,全部其他 Thread 通过,继续执行或结束。
所有到达 CyclicBarrier 的屏障点 point 时候,该屏障点 point 结束,继续执行。如果不能全部到达(中断、失败或者超时等原因),设置 BrokenBarrierException 引起所有调用 await 方法的“Thread”全部报错,全部要么全不 (all-or-none) 的破坏模式。
reset 也是抛出 BrokenBarrierException。
转自:http://www.cnblogs.com/lsx1993/p/4631733.html
【CountDownLatch、CyclicBarrier 和 Semaphore】
http://www.cnblogs.com/dolphin0520/p/3920397.html
【CountDownLatch 同步工具类】
http://www.importnew.com/15731.html
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于