Java并发——AQS

2018-08-10  本文已影响0人  sortinnauto

AQS,即AbstractQuenedSynchronizer,顾名思义为抽象的队列式同步器。AQS定义了一套多线程访问共享资源的的同步器框架,许多同步类的实现都是依靠这个框架,比如谈并发就一定会谈的ReentrantLock、Semaphore、CountDownLatch等。它是Java并发包(java.util.concurrent)里实现锁、同步的一个重要基础框架。


AQS类维护了一个wait quene,这个FIFO的等待队列是CLH锁队列的变体(a variant of a "CLH" lock queue)。CLH锁通常用于自旋锁(spinlocks)。而在AQS中,CLH锁被用来阻塞同步器。但是呢,即使作用有变,其基本的策略(tactic)是相同的,即在节点(Node)的前驱节点(predecessor)中保存一些关于线程的控制信息。每个节点中的 status 字段跟踪(trace)线程是否应该被阻塞。当一个节点的前驱节点被释放时,前驱节点会通知(signal)当前节点。

AQS 中的队列是由 Node 节点组成的双向链表实现的,那 Node 到达是什么呢?其实 Node 是 AQS的一个内部类,它是对访问同步资源的线程的封装,可以简单理解为一个 Node 代表一个需要同步的线程及其状态。

AQS定义了两种资源共享的方式:独占式( Exclusive ,只能一个线程独享,如 ReentrantLock )和共享式( Shared ,多个线程可同时执行,如 Semaphore / CountDownLatch )。

两种资源共享方式

每个 Node 都有其线程等待状态, Node 类维护一个 waitStatus 变量来表明这些状态:


waitStatus的取值 waitStatus

写入等待队列 addWaiter()

入队 enq( Node )

aquire( int )

独占模式下获取资源

这个方法是在独占模式下线程获取共享资源的顶级入口,如果获取到资源,则返回,否则线程进入等待队列,知道获取到资源为止。这个方法忽略中断,可以用来实现 Lock 接口的 lock 方法。

这个方法的流程如下:

  1. tryAcquire()尝试直接去获取资源,如果成功则直接返回;

  2. addWaiter()将该线程加入等待队列的尾部,并标记为独占模式;

  3. acquireQueued()使线程在等待队列中获取资源,一直获取到资源后才返回。如果在整个等待过程中被中断过,则返回true,否则返回false。

  4. 如果线程在等待过程中被中断过,它是不响应的。只是获取资源后才再进行自我中断selfInterrupt(),将中断补上。


aquire() 方法的传入参数先辈传给 tryAcquire( int ) 方法,这个方法如下:

tryAcquire( int )

tryAcquire( int )

这个方法体其实没有提供实现,只是给出了一异常,这也是 AQS 被当做一个框架的原因,具体的资源获取方式交给自定义的同步器去实现。现在我们只需知道这个方式是在尝试获取资源。

acquireQuened( Node, int )

acquireQuened( Node, int )

aquire() 方法中,如果尝试获取资源失败,那么就将当前线程放入队列中。acquireQuened() 的作用就是在队列中等待被唤醒,直到被唤醒,也是在做“自旋”。

未完待续

上一篇 下一篇

猜你喜欢

热点阅读