1.AQS同步器是什么?用来做什么?怎么自定义同步器?
2.AQS同步器是怎么实现的?
3.AQS同步器访问资源有哪些方式?用AQS同步器实现自定义的同步器应该怎么做?
4.AQS同步器组件有哪些?
1.Q:AQS同步器是什么?用来做什么?怎么自定义同步器?
A:
AQS同步器是AbstractQueuedSynchronizer,抽象队列同步器。它在java.util.concurrent.locks包下。
它是用来针对具体需求自定义同步器的框架。ReentrantLock,Semaphore(信号量),CyclicBarrier(循环屏障,或叫循环栅栏)都是基于AQS来实现的。
只要继承该类,重写独占资源方式方法(tryAcquire获取、tryRelease释放)或共享资源方式方法(tryAcquireShared获取、tryReleaseShared释放)。就可以根据需求自定义同步器。
2.Q:AQS同步器是怎么实现的?
A:
AQS实现思路/方式是:
类比ReentrantLock的实现。
当资源是空闲的时候,某线程请求该资源,该资源把该线程标记为工作线程,该线程把该资源锁定。
当有其他线程请求该资源时,把其他线程放入CLH队列中等待(CLH队列是一个虚拟的双向队列,也就是没有实体的队列,但是各线程结点之间有关联关系)
然后AQS同步器(是个java类)内部有个state的int类型变量,来表示同步状态。并用CAS操作来修改这个变量。
比如ReentrantLock,state为0表示还没有线程获得该资源。当有线程获得该资源的时候state加1,由于是可重入锁,所以该线程可以再次获得该资源,state再加1。释放一次,state就减1。直到state为0,资源被该线程释放完毕。
3.Q:AQS同步器访问资源有哪些方式?用AQS同步器实现自定义的同步器应该怎么做?
A:AQS同步器访问资源有独占方式和共享方式两种。
AQS同步器中线程都是在CLH队列中的。
独占方式:就是资源只能被一个线程使用。如ReentrantLock
对于排在CLH队列中的线程,有两种锁:
公平锁:先来的进程先获得资源。
非公平锁:先来的进程不一定先获得资源,谁竞争到资源就是谁的。
共享方式:就是资源可以被多个线程共同使用。如Semaphore信号量,CountDownLatch倒计时器,CyclicBarrier循环栅栏。
而读写锁,既支持独占方式,又支持共享方式。
实现自定义AQS同步器时,只需要继承AbstractQueuedSynchronized类,并根据需要重写tryAcquire()和tryRelease()——以独占方式,或者重写tryAcquireShared()和tryReleaseShared()——以共享方式。
这些方法称为模板方法。源码里默认这些方法都抛出一个异常,所以才需要继承该类来重写这些方法。
这样就行,像那些线程在CLH队列中的一些细节,AQS同步器已经帮我们实现了,不需要操心。
4.Q:AQS同步器组件有哪些?
A:
Semaphore信号量:资源可以被多个线程共享使用。
CountDownLatch倒计时器:它可以让线程一直等待,直到倒计时结束时,再让线程运行。
CyclicBarrier循环栅栏:当一些线程先到达该栅栏时先让他们等待,直到最后一个线程到达该栅栏时,才让所有线程工作。它可以实现当多个线程同时运行的效果
CountDownLatch倒计时器和CyclicBarrier循环栅栏都是让先来的线程等待,来达到各自的效果的。
这些AQS同步组件都是共享方式的,而ReentrantLock是独占方式的。