CountDownLatch
和CyclicBarrier
是jdk concurrent包下非常有用的两个并发工具类,它们提供了一种控制并发流程的手段。
CountDownLatch
CountDownLatch
允许一个或多个线程等待其他线程完成操作。
CountDownLatch
可以把它看作一个计数器,只不过这个计数器的操作是原子操作,同时只能有一个线程去操作这个计数器,也就是同时只能有一个线程去减这个计数器里面的值。
应用场景
有一个任务想要往下执行,但必须要等到其他的任务执行完毕后才可以继续往下执行。假如我们这个想要继续往下执行的任务调用一个CountDownLatch
对象的await()
方法,其他的任务执行完自己的任务后调用同一个CountDownLatch
对象上的countDown()
方法则计数减1,这个调用await()
方法的任务将一直阻塞等待,直到这个CountDownLatch
对象的计数值减到0为止。
案例
比如:有2个工人在为老板干活,当2个工人把一天的活都干完了的时候,老板就来检查所有工人所干的活。记住这个条件:三个工人先全部干完活,老板才检查。所以在这里设计两个类,Worker代表工人,Boss代表老板。代码如下:
Worker.java
|
|
Boss.java
|
|
CountDownLatchTest.java
|
|
CountDownLatch源码分析
- 自定义同步器
Sync
实现
|
|
- 构造方法
|
|
从构造方法的具体实现可以看出,通过构造方法传入的int型参数count其实就是同步器的状态。
countDown
方法
|
|
整个countDown
只做了一件事情,释放同步状态,同步状态在这里的实际意义也就是需要等待的完成的点的数量,只要每完成一个点,就调用countDown方法释放同步状态。
await
方法
CountDownLatch
提供带超时时间的await
和不带超时时间的await
:
|
|
await
的实质是在获取同步状态,同步状态state == 0成立,当前等待完成的点均已完成,主线程继续往下执行,否则,主线程进入等待队列自旋等待直到同步状态释放后state == 0。有些时候主线程是不能一直自旋等待,这个时候带超时时间的await就派上用场了,设置超时时间,如果在指定时间内N个点都未完成,返回false,主线程不再等待,继续往下执行。
CountDownLatch总结
CountDownLatch
实质上就是一个AQS
计数器,通过AQS
来实现线程的等待与唤醒。