AQS

2021-01-20  本文已影响0人  得力小泡泡

AQS官方解读

AQS 只是一个框架,具体资源的获取/释放方式交由自定义同步器去实现,AQS 这里只定义了一个接口,具体资源的获取交由自定义同步器去实现了

提供了一个框架用来实现阻塞的锁和相关的同步器(比如信号量,事件等等),他们依赖于FIFO的等待队列(相对于synchronized, 它在底层其实也是对应两个集合:WaitSet,EntrySet,而AQS只是它又做了一些扩展,等待队列有多个,而阻塞队列只有一个)。该类被设计成用于为大多数依赖单个int值表示状态的同步器提供有用的基础(其中提到了一个int值,这个值的含义非常的丰富,不同的场景下它代表的意思也是截然不同的,后面会说)。子类必须定义改变此状态的受保护方法,以及根据该对象被获取或释放来定义该状态的含义

其中它定义的受保护的方法,而且它里面都是直接抛异常了,很明显具体类是必须要定义这些受保护的方法的,为啥?因为抽象的父类是没法实现这些方法,因为具体的场景需要依赖于子类。


image.png
image.png

鉴于这些,给定这些,这个类中的其他方法执行所有排队和阻塞机制。子类可以维护其他的状态字段,但是仅跟踪使用getState(),setState(int) 和 compareAndSetState(int, int)方法操纵的以原子方式更新的int值进行同步

总的来说,它是提供了一个底层的基础组件,至于它到底是在什么场景下有什么样的表述形式是由具体子类来实现的。

AQS抽象类核心抽象方法

定义了若干个protected的重要核心方法,而这些受保护的方法它具体逻辑的执行应该取决于它的子类它所处理的具体业务的场景以及子类所代表的具体的业务的含义

里面的实现直接抛出异常了,很明显就是子类是一定要来实现它的

    protected boolean tryAcquire(int arg) {
        throw new UnsupportedOperationException();
    }

1、尝试以独占模式获取。 该方法应该查询对象的状态是否允许以独占模式获取,如果是,则获取它。

    protected boolean tryRelease(int arg) {
        throw new UnsupportedOperationException();
    }

2、尝试去设置AQS中的state用来反映在排它模式下的一次释放,此方法经常是被线程调用执行锁的释放

    protected int tryAcquireShared(int arg) {
        throw new UnsupportedOperationException();
    }

3、尝试以共享模式获取。 该方法应该查询对象的状态是否允许在共享模式下获取该对象,如果是这样,就可以获取它。

    protected boolean tryReleaseShared(int arg) {
        throw new UnsupportedOperationException();
    }

4、尝试去设置AQS中的state用来反映在共享模式下的一次释放,此方法经常是被线程调用执行锁的释放

    protected boolean isHeldExclusively() {
        throw new UnsupportedOperationException();
    }

5、如果同步锁以独占的方式被当前调用者线程所持有则返回true

如果是在独占模式下,必须实现tryAcquire()和tryRelease()方法;
如果是在共享模式下,必须实现tryAcquireShared()和tryReleaseShared()方法

AQS结构图

image.png
AQS中有两个内部类
流程

3、论AQS类中的state变量

image.png

在AQS当中有一个重要的成员变量来决定当前的线程是否有能力能够拿到AQS执行的资源,在不同的场景下其state的含义是不一样的,下面举两个场景,先有个大概印象:

关于这个变量是AQS设计最为精妙的地方,该字段的含义得从它的子类进行了解。也就是说当调用了AQS中的某一个condition.await()之后得依据这个state变量来决定线程是否能执行AQS的资源,如果不能则就会进入FIFO里面进行阻塞等待了。

4、AQS在Java并发包的应用规律

在子类中,该子类运用了AQS的机制,该子类中都定义了一个Sync内部类,该类是继承于AbstractQueuedSynchronizer,然后具体由它来实现AQS的细节

可重入锁


image.png

读写锁


image.png

5、Node的结构,以及Node在FIFO阻塞队列和ConditionObject等待队列的使用方式

Node的结构

static final class Node {
    static final Node SHARED = new Node();//共享锁结点
    static final Node EXCLUSIVE = null;//排他锁结点
    volatile Node prev;//使用在FIFO阻塞队列中指向前驱结点
    volatile Node next;//使用在FIFO阻塞队列中指向后继结点
    Node nextWaiter;//使用在ConditionObject中指向下一个结点
    volatile Thread thread;//当前线程
}

从源码可以说明,ConditionObject等待队列是使用单向链表实现的,而FIFO阻塞队列是使用双向链表实现的

(1)FIFO阻塞队列添加元素时,使用的是Node的prev和next实现前后指针

    private Node addWaiter(Node mode) {
        Node node = new Node(Thread.currentThread(), mode);
        // Try the fast path of enq; backup to full enq on failure
        Node pred = tail;
        if (pred != null) {
            node.prev = pred;
            if (compareAndSetTail(pred, node)) {
                pred.next = node;
                return node;
            }
        }
        enq(node);
        return node;
    }

(2)ConditionObject等待队列添加元素时,使用的是Node的nextWaiter实现后继指针

    private Node addConditionWaiter() {
        Node t = lastWaiter;
        // If lastWaiter is cancelled, clean out.
        if (t != null && t.waitStatus != Node.CONDITION) {
            unlinkCancelledWaiters();
            t = lastWaiter;
        }
        Node node = new Node(Thread.currentThread(), Node.CONDITION);
        if (t == null)
            firstWaiter = node;
        else
            t.nextWaiter = node;
        lastWaiter = node;
        return node;
    }

6、关于AQS与synchronized关键字之间的关系

1、synchronized关键字在底层C++实现中,存在两个重要的数据结构(集合):WaitSet,EntryList,都是用环形双向链表构造成的
2、WaitSet中存放的是调用了Object的wait方法的线程对象(被封装成C++的Node对象)
3、EntryList中存放的是陷入到阻塞状态,需要获取monitor的那些线程对象
4、当一个线程被notify后,它就会从WaitSet中移动到EntryList中
5、进入到EntryList后,该线程依然需要与其他线程争抢monitor对象
6、如果争抢到,就表示该线程获取到了对象的锁,它就可以排他方式执行对应的同步代码块

1、AQS中存在两种队列,分别是Condition对象上的条件队列,以及AQS本身的阻塞队列
2、这两个队列中的每一个对象都是Node实例(封装了线程对象)
3、当位于Condition条件队列中的线程被其他线程singal后,分两种情况,ReentrantLock 和 ReentrantReadWriteLock 同时都有公平锁与非公平锁两种情况,具体看ReentrantLock 和 ReentrantReadWriteLock两篇文章
4、AQS的FIFO阻塞队列本质上是由一个双向链表构成的,Condition的等待队列本质上是由一个单向链表构成的,
5、在获取AQS锁时,这些进入到FIFO阻塞队列中的线程会按照在队列中的排序先后获取
6、当AQS阻塞队列中的线程获取到锁后,就表示该线程已经可以正常执行了
7、陷入到阻塞状态的线程,依然需要进入到操作系统的内核态,进入阻塞(park方法实现)

上一篇 下一篇

猜你喜欢

热点阅读