CountDownLatch同步组件之初探

前序:此文目前仅是半成品,后期完全读透再更文。

        一、因为突发奇想,以前做小程序中遇到的一个问题。页面展示时,初始化需要异步接口返回数据,才能进行下一步页面渲染工作。当然你完全可以使用一个同步接口,但同步的话可能会出现其它的问题。而且那个时候,我用的第三方API也没有提供同步的API接口。简化点我提的问题,归根是:页面展示为主线程,异步接口调用为子线程。问题:主线程如何实现等所有子线程调用完成后,再往下执行呢?

        二、这个实现有很多。join()可以、wait()可以,还有JUC包(java.util.concurrent)。接下来先生要介绍的就是JUC包中的一个同步组件CountDownLatch。(必须得艾特一下这个包的开发者:@author Doug Lea大佬)


      三、CountDownLatch【闭锁】【共享锁】

        抛去底层实现,跟我们使用息息相关的是它类中的成员变量以及方法。(还是第一次读源码,原谅我这不正宗的Javaer)

CountDownLatch同步组件之初探_第1张图片
CountDownLatch类

        3.1、直观可见:CountDownLatch包含一个内部类Sync类,Sync类继承AbstractQueuedSynchronizer类(也就是传说中的AQS,同步队列器),想要深入了解,当然得继续剖析这两个类。最后会提供相应传送门链接得,莫急莫急。

        3.2、成员变量:Sync sync;

            当然是提供方法,供CountDownLatch调用,实现同步的。

        3.3、成员方法:       

            public CountDownLatch(int count);    初始化三个类,并通过传参count来设置AQS中state变量。

            public void countDown();    通过调用sync.releaseShared()释放共享锁资源,通过变量state减一的方式,释放线程。当state==0时,即队列中资源为空。

            public void await();    调用该方法的当前线程均被挂起,技术用到了LockSupport.park(this),在此不拓展。直到队列资源为空,或是被中断时,await被挂起的线程被唤醒。

            public boolean await(long timeout, TimeUnit unit);    在等待特定时间后,就会不再阻塞当前线程,所有线程继续并行竞争。

    四、传送门:

        4.1、CountDownLatch简介.

        4.2、AbstractQueuedSynchronizer(同步队列器).上部分简介    下部分简介

        4.3、CAS很基础的入门.


    五、Demo展示:

CountDownLatch同步组件之初探_第2张图片
TestCountDownLatch类

        5.1、使用场景只能是等所用线程执行完才能,才能唤醒被await()挂起的线程嘛?

            当然不是的。仔细思考一下,CountDownLatch其实依靠变量state==0,来控制是否唤醒被挂起的线程。所以,我们使用场景还可以是:只要所有线程都开始运行了,就唤醒被挂起的线程。举个例:比如车间有三个流水线,分别是A生产左裤脚、B生产右裤脚、C合成一条裤子。C可以一开始是挂起的,但只要A和B已经投入生产了,那么C也应该运行起来。代码中,我们只要改一下countDown()调用的位置即可,想必聪明机智如你应该想到了如何修改了吧。

        5.2、await()不仅仅使用在主线程吧?

            对的,你当然可以使用在子线程,还可以多次使用讷。当多次使用await,要记得被挂起的线程,只有在state==0,也就是countDown()调用的次数,要等于构造函数的初始值。否则,你等着被老板喷吧。使用场景:还是那个工厂,由于工厂最初的建造,有了十个车间。但是由于经济不景气,车间五个就够啦。那么剩下的车间完全可以挂起。


    我是先生,找寻着那位迷路的Miss。最后,愿各位javaer,合上电脑的刹那,有着侠客收剑入鞘的骄傲!

你可能感兴趣的:(CountDownLatch同步组件之初探)