一、AQS的介绍
队列同步器AbstractQueuedSynchronizer(AQS)是构建锁或者其他同步组件的基础框架,是实现Lock的基础。它使用了一个volatile修饰的int变量来表示同步状态,并维护了一个FIFO队列来完成资源获取线程的排队。
public abstract class AbstractQueuedSynchronizer extends AbstractOwnableSynchronizer
implements java.io.Serializable {
private transient volatile Node head;//头节点
private transient volatile Node tail;//尾节点
private volatile int state; //同步状态
protected final int getState() {
return state;
}
protected final void setState(int newState) {
state = newState;
}
protected final boolean compareAndSetState(int expect, int update) {
return unsafe.compareAndSwapInt(this, stateOffset, expect, update);
}
...
从上面AQS的部分代码可以看到,AQS是一个类,它包含了表示同步状态的state变量(volatile修饰);维护队列的两个引用头节点head和尾节点tail(volatile修饰);以及提供了三个主要方法,用来保证同步状态的改变是线程安全的。
那么AQS是如何实现锁的呢?
当我们需要实现锁的时候,首先继承AQS并重写指定的方法,然后将AQS子类组合在自定义组件(锁)的实现中,并调用AQS的模板方法,而这些模板方法将会调用我们重写的方法(模板方法模式),这就可以达到我们想要的效果。
注:重写指定的方法时需要用到AQS中的三个主要方法来对同步状态进行访问或修改。
AQS中可重写的方法如下:
protected boolean tryAcquire(int arg) {} //独占式获取
protected boolean tryRelease(int arg) {} //独占式释放
protected int tryAcquireShared(int arg) {} //共享式获取
protected boolean tryReleaseShared(int arg) {} //共享式释放
protected boolean isHeldExclusively() {} //判断AQS是否被该线程独占
来看一个独占锁的示例。
class Mutex implements Lock, java.io.Serializable {
// 内部类,自定义同步器,继承AQS
private static class Sync extends AbstractQueuedSynchronizer {
// 重写方法——是否处于占用状态
protected boolean isH