AQS 框架论文翻译 —— The java.util.concurrent Synchronizer Framework

Java 并发原理系列(二)

Posted by John Mactavish on April 10, 2021

原论文地址:http://gee.cs.oswego.edu/dl/papers/aqs.pdf

这是了解 JUC 的 AbstractQueuedSynchronizer 必读的论文,论文作者是 JUC 设计者————大名鼎鼎的 Doug Lea。 文中讨论了 AQS 的设计目的、思路与实现,不仅对于了解 JUC 有作用,还有助于体会程序设计的思想。本来想自己翻译的, 但是惊喜地发现了这篇译文,特此转载。

翻译转载自:https://blog.csdn.net/mar_ljh/article/details/105531003

译者:赣江

本文在转载基础上稍有修改与润色。文中的“注”是我添加的,而“译者注”是这位译者添加的。

J2SE 1.5java.util.concurrent 包(下称 j.u.c 包)中,大部分的同步器(例如锁 locks,屏障 barriers,等等)都是基于AbstractQueuedSynchronizer 类(下称 AQS 类)这个小的框架。该框架提供了一种通用的机制来实现管理原子性状态(atomically managing synchronization state)、线程的阻塞(blocking)和非阻塞(unblocking)以及队列(queuing)。这篇论文主要描述了这个的框架基本原理、设计、实现、用法以及性能。

1 背景介绍

通过 JCPJSR166 规范,Java1.5 版本引入了 j.u.c 包,这个包提供了一系列支持中等程度并发的类。这些组件是一系列的同步器(抽象数据类型(ADT))。这些同步器主要维护着以下几个功能:内部同步状态(synchronization state)的管理(例如:表示一个锁的状态是获取还是释放),同步状态的更新和检查操作,且至少有一个方法会导致调用线程在同步状态被获取时阻塞,以及在其他线程改变这个同步状态时解除线程的阻塞。上述的这些的实际例子包括:互斥排它锁的不同形式、读写锁、信号量、屏障、Future、事件指示器以及传送队列等。

几乎任一同步器都可以用来实现其他形式的同步器。例如,可以用可重入锁实现信号量或者用信号量实现可重入锁。但是,这样做带来的复杂性、开销、不灵活使其至多只能是个二流工程,且缺乏吸引力。如果任何类似的构造方式不能在本质上比其他形式更简洁,那么开发者就不应该随意地选择其中的某个来构建另一个同步器。取而代之,JSR166 建立了一个小框架,AQS 类。这个框架为构造同步器提供一种通用的机制,并且被 j.u.c 包中大部分类使用,同时很多用户也用它来定义自己的同步器。

在这篇论文剩下来的部分会讨论这个框架的需求、设计与实现背后的主要思路、示例用法,以及性能指标的一些测量。

2 需求

2.1 功能

同步器一般包含两种方法,一种是 acquire,另一种是 releaseacquire 操作阻塞调用的线程,直到或除非同步状态允许其继续执行。而release 操作则是通过某种方式改变同步状态,使得一或多个被 acquire 阻塞的线程继续执行。

j.u.c 包中并没有对同步器的 API 做一个统一的定义。因此,有一些类定义了通用的接口(如 Lock),而另外一些则定义了其专有的版本。因此在不同的类中,acquirerelease 操作的名字和形式会各有不同。例如:Lock.lockSemaphore.acquireCountDownLatch.awaitFutureTask.get,在这个框架里,这些方法都是 acquire 操作。但是,j.u.c为支持一系列常见的使用选项,在类间都有个一致约定。在有意义的情况下,每一个同步器都支持下面的操作:

  • 阻塞和非阻塞(例如 tryLock)同步
  • 可选的超时设置,让调用者可以放弃等待
  • 通过中断实现的任务取消,通常是分为两个版本:一个 acquire 可取消,而另一个不可以

同步器的实现根据其状态是否独占而有所不同。独占状态的同步器,在同一时间只有一个线程可以通过阻塞点,而共享状态的同步器可以同时有多个线程在执行。一般锁的实现类往往只维护独占状态,但是,例如计数信号量在数量许可的情况下,允许多个线程同时执行。为了使框架能得到广泛应用,这两种模式都要支持。

j.u.c 包里还定义了 Condition 接口,用于支持监控形式的 await/signal 操作,这些操作与独占模式的 Lock 类有关,且 Condition 的实现天生就和与其关联的 Lock 类紧密相关。

2.2 性能目标

Java 内置锁(使用 synchronized 的方法或代码块)的性能问题一直以来引人关注,并且已经有一系列的文章讨论其构造(例如引文13)。然而,大部分的研究主要关注的是在单核处理器、单线程环境中,如何尽量降低其空间(因为任何的 Java 对象都可以当成是锁)和时间的开销。对于同步器来说这些都不是特别重要:程序员仅在需要的时候才会使用同步器,因此并不需要压缩空间来避免浪费,并且同步器几乎是专门用在多线程设计中(特别是在多核处理器上),在这种环境下,偶尔的竞争是在意料之中的。而常规的 JVM 锁优化策略主要是针对零竞争的场景,其它场景则使用缺乏可预见性的“慢速路径(slow paths)”(注:指的应该是偏向锁的负优化,见系列上一篇文章),所以常规的 JVM 锁优化策略并不适用于严重依赖于 j.u.c 包的典型的多线程服务端应用。

这里主要的性能目标是可伸缩性,即在大部分情况下,即使,或特别在同步器有竞争的情况下,稳定地保证其效率。在理想的情况下,不管有多少线程正试图通过同步点,通过同步点的开销都应该是个常量。在某一线程被允许通过同步点但还没有通过的情况下,使其耗费的总时间最少,这是主要目标之一。然而,这也必须考虑平衡各种资源,包括总 CPU 时间的需求,内存负载以及线程调度的开销。例如:获取自旋锁通常比阻塞锁所需的时间更短,但是通常也会浪费 CPU 时钟周期,并且造成内存竞争,所以使用的并不频繁。(注:系列上一篇文章也讨论过自旋 CAS 与阻塞系统调用的权衡(trade-off)

同步器的上述目标适用于以下两种不同的使用类型。对于大部分应用程序,要求最大化其总的吞吐量、容错性,并且最好保证尽量减少饥饿的情况;而对于那些控制资源分配的程序来说,更重要的是去维持多线程的公平性,相对可以接受较差的总吞吐量。没有任何框架可以为用户决定应该选择哪一种类型,因此,两种不同的公平策略都应该提供。

无论同步器的内部实现是多么的精雕细琢,它还是会在某些应用中产生性能瓶颈。因此,框架必须提供相应的监视工具让用户发现和缓和这些瓶颈。至少需要提供一种方式来确定有多少线程被阻塞了。

3 设计与实现

同步器背后的基本思想非常简单。acquire 操作如下:

while (synchronization state does not allow acquire) {  
    enqueue current thread if not already queued;
    possibly block current thread;
}
dequeue current thread if it was queued;

release 操作如下:

update synchronization state;
if (state may permit a blocked thread to acquire)   
    unblock one or more queued threads;

为了实现上述操作,需要下面三个基本组件的相互协作:

  • 同步状态的原子性管理;
  • 线程的阻塞与解除阻塞;
  • 队列的管理;

创建一个框架分别实现这三个组件是有可能的。但是,这会让整个框架既难用又没效率。例如:存储在队列节点的信息必须与解除阻塞所需要的信息一致,而暴露出的方法的签名必须依赖于同步状态的特性。

同步器框架的核心决策是为这三个组件选择一个具体实现,同时在使用方式上又有大量选项可用。这里有意地限制了其适用范围,但是提供了足够的效率,使得实际上没有理由在合适的情况下不用这个框架而去重新建造一个。

3.1 同步状态

AQS 类使用单个 int(32 位) 来保存同步状态,并暴露出 getStatesetState 以及 compareAndSet 操作来读取和更新这个状态。这些方法都依赖于 j.u.c.atomic 包的支持,这个包提供了在读和写上兼容 JSR133volatile 语义的原子类,并且通过使用本地的 compare-and-swapload-linked/store-conditional 指令来实现compareAndSetState,使得仅当同步状态拥有一个期望值的时候,才会被原子地设置成新值。

将同步状态限制为一个 32 位 的整型是出于实践上的考量。虽然 JSR166 也提供了 64 位 long 字段的原子性操作,但这些操作在很多平台上还是使用内部锁的方式来模拟实现的,这使同步器的性能可能不会很理想。当然,将来可能会有一个类是专门使用 64位 的状态的。然而现在就引入这么一个类到这个包里并不是一个很好的决定(译者注:JDK1.6 中已经包含java.util.concurrent.locks.AbstractQueuedLongSynchronizer 类,即用 long 形式维护同步状态的一个 AbstractQueuedSynchronizer 版本)。目前来说,32 位的状态对大多数应用程序都是足够的。在 j.u.c 包中,只有一个同步器类可能需要多于 32位 来维持状态,那就是 CyclicBarrier 类,所以,它用了锁(该包中大多数更高层次的工具亦是如此)。

基于 AQS 的具体实现类必须根据暴露出的状态相关的方法定义 tryAcquiretryRelease 方法,以控制 acquirerelease 操作。(注:这是设计模式中的模板方法 (Template method pattern))当同步状态满足时,tryAcquire 方法必须返回 true,而当新的同步状态允许后续 acquire 时,tryRelease 方法也必须返回 true。这些方法都接受一个 int 类型的参数用于传递想要的状态。例如:可重入锁中,当某个线程从条件等待中返回,然后重新获取锁时,重新建立循环计数的场景。很多同步器并不需要这样一个参数,因此忽略它即可。

3.2 阻塞

JSR166 之前,阻塞线程和解除线程阻塞都是基于 Java 内置监视器,没有 Java API 可以用来创建同步器。唯一可以选择的是 Thread.suspendThread.resume,但是它们都有无法解决的竞态问题,所以也没法用:当一个非阻塞的线程在一个正准备阻塞的线程调用 suspend 前调用了 resume,这个 resume 操作将不会有什么效果。

j.u.c 包有一个 LockSuport 类,这个类中包含了解决这个问题的方法。方法 LockSupport.park 阻塞当前线程除非/直到有个 LockSupport.unpark 方法被调用(unpark 方法被提前调用也是可以的)。unpark 的调用是没有被计数的,因此在一个 park 调用前多次调用 unpark 方法只会解除一个 park 操作。另外,它们作用于每个线程而不是每个同步器。一个线程在一个新的同步器上调用 park 操作可能会立即返回,因为在此之前可能有“剩余的” unpark 操作。但是,在缺少一个 unpark 操作时,下一次调用 park 就会阻塞。虽然可以显式地消除这个状态(译者注:就是多余的 unpark 调用),但并不值得这样做。在需要的时候多次调用 park 会更高效。

这个简单的机制与有些用法在某种程度上是相似的,例如 Solaris-9 的线程库,WIN32 中的“可消费事件”,以及 Linux 中的 NPTL 线程库。因此最常见的运行 Java 的平台上都有相对应的有效实现。(但目前 SolarisLinux 上的 Sun Hotspot JVM 参考实现实际上是使用一个 pthreadcondvar 来适应目前的运行时设计的)。park 方法同样支持可选的相对或绝对的超时设置,以及与 JVMThread.interrupt 结合 —— 可通过中断来 unpark 一个线程。

3.3 队列

整个框架的关键就是如何管理被阻塞的线程的队列,该队列是严格的 FIFO 队列,因此,框架不支持基于优先级的同步。

同步队列的最佳选择是自身没有使用底层锁来构造的非阻塞数据结构,目前,业界对此很少有争议。而其中主要有两个选择:一个是 Mellor-Crummey 和 Scott 锁(MCS 锁)9 的变体,另一个是 Craig,Landin 和 Hagersten 锁(CLH 锁)5810 的变体。一直以来,CLH 锁仅被用于自旋锁。但是,在这个框架中,CLH 锁显然比 MCS 锁更合适。因为 CLH 锁可以更容易地去实现“取消(cancellation)”和“超时”功能,因此我们选择了 CLH 锁作为实现的基础。但是最终的设计已经与原来的 CLH 锁有较大的出入,因此下文将对此做出解释。

CLH 队列实际上并不那么像队列,因为它的入队和出队操作都与它的用途(即用作锁)紧密相关。它是一个链表队列,通过两个字段 headtail 来存取,这两个字段是可原子更新的,两者在初始化时都指向了一个空节点。

clh

一个新的节点,node,通过一个原子操作入队:

do {
    pred = tail;
} while(!tail.compareAndSet(pred, node));

每一个节点的“释放”状态都保存在其前驱节点中。因此,自旋锁的“自旋”操作就如下:

while (pred.status != RELEASED); // spin

自旋后的出队操作只需将 head 字段指向刚刚得到锁的节点:

head = node;

CLH 锁的优点在于其入队和出队操作是快速、无锁的,以及无障碍的(即使在竞争下,某个线程总会赢得一次插入机会而能继续执行); 且探测是否有线程正在等待也很快(只要测试一下 head 是否与 tail 相等);同时,“释放”状态是分散的(译者注:几乎每个节点都保存了这个状态,当前节点保存了其后驱节点的“释放”状态,因此它们是分散的,不是集中于一块的。),避免了一些不必要的内存竞争。

在原始版本的 CLH 锁中,节点间甚至都没有互相链接。自旋锁中,pred 变量可以是一个局部变量。然而,ScottScherer 证明了通过在节点中显式地维护前驱节点,CLH 锁就可以处理“超时”和各种形式的“取消”:如果一个节点的前驱节点取消了,这个节点就可以滑动去使用前面一个节点的状态字段。

为了将 CLH 队列用于阻塞式同步器,需要做些额外的修改以提供一种高效的方式定位某个节点的后继节点。在自旋锁中,一个节点只需要改变其状态,下一次自旋中其后继节点就能注意到这个改变,所以节点间的链接并不是必须的。但在阻塞式同步器中,一个节点需要显式地唤醒(unpark)其后继节点。

AQS 队列的节点包含一个 next 链接到它的后继节点。但是,由于没有针对双向链表节点的类似 compareAndSet 的原子性无锁插入指令,因此这个 next 链接的设置并非作为原子性插入操作的一部分,而仅是在节点被插入后简单地赋值:

pred.next = node;

第二个对 CLH 队列主要的修改是将每个节点都有的状态字段用于控制阻塞而非自旋。在同步器框架中,仅在线程调用具体子类中的 tryAcquire 方法返回 true 时,队列中的线程才能从 acquire 操作中返回;而单个“released”位是不够的。但仍然需要做些控制以确保当一个活动的线程位于队列头部时,仅允许其调用 tryAcquire;这时的 acquire 可能会失败,然后(重新)阻塞。这种情况不需要读取状态标识,因为可以通过检查当前节点的前驱是否为 head 来确定权限。与自旋锁不同,读取 head 以保证复制时不会有太多的内存竞争( there is not enough memory contention reading head to warrant replication.)。然而,“取消”状态必须存在于状态字段中。next 链接仅是一种优化。如果通过某个节点的 next 字段发现其后继结点不存在(或看似被取消了),总是可以使用 pred 字段从尾部开始向前遍历来检查是否真的有后续节点。

队列节点的状态字段也用于避免没有必要的 parkunpark 调用。虽然这些方法跟阻塞原语一样快,但在跨越 JavaJVM runtime 以及操作系统边界时仍有可避免的开销。在调用 park 前,线程设置一个“唤醒(signal me)”位,然后再一次检查同步和节点状态。一个释放的线程会清空其自身状态。这样线程就不必频繁地尝试阻塞,特别是在锁相关的类中,这样会浪费时间等待下一个符合条件的线程去申请锁,从而加剧其它竞争的影响。除非后继节点设置了“唤醒”位(译者注:源码中为 -1),否则这也可避免正在 release 的线程去判断其后继节点。这反过来也消除了这些情形:除非“唤醒”与“取消”同时发生,否则必须遍历多个节点来处理一个似乎为 nullnext 字段。

同步框架中使用的 CLH 锁的变体与其他语言中的相比,主要区别可能是同步框架中使用的 CLH 锁需要依赖垃圾回收管理节点的内存,这就避免了一些复杂性和开销。但是,即使依赖 GC 也仍然需要在确定链接字段不再需要时将其置为 null。这往往可以与出队操作一起完成。否则,无用的节点仍然被引用,它们就没法被回收。(注:参考 JVM GC 算法

其它一些更深入的微调,包括 CLH 队列首次遇到竞争时才需要的初始空节点的延迟初始化等,都可以在 J2SE1.5 的版本的源代码文档中找到相应的描述。

抛开这些细节,基本的 acquire 操作的最终实现的一般形式如下(互斥,非中断,无超时):

if(!tryAcquire(arg)) {
    node = create and enqueue new node;
    pred = node's effective predecessor;
    while (pred is not head node || !tryAcquire(arg)) {
        if (pred's signal bit is set)
            park()
        else
            compareAndSet pred's signal bit to true;
        pred = node's effective predecessor;
    }
    head = node;
}

release 操作:

if(tryRelease(arg) && head node's signal bit is set) {
    compareAndSet head's bit to false;
    unpark head's successor, if one exist
}

acquire 操作的主循环次数依赖于具体实现类中 tryAcquire 的实现方式。另一方面,在没有“取消”操作的情况下,每一个组件的 acquirerelease 都是一个 O(1) 的操作,忽略 park 中发生的所有操作系统线程调度。

支持“取消”操作主要是要在 acquire 循环里的 park 返回时检查中断或超时。由超时或中断而被取消等待的线程会设置其节点状态,然后 unpark 其后继节点。在有“取消”的情况下,判断其前驱节点和后继节点以及重置状态可能需要 O(n) 的遍历(n 是队列的长度)。由于“取消”操作,该线程再也不会被阻塞,节点的链接和状态字段可以被快速重建。

3.4 条件队列

AQS 框架提供了一个 ConditionObject 类,给维护独占同步的类以及实现 Lock 接口的类使用。一个锁对象可以关联任意数目的条件对象,可以提供典型的管程风格的 awaitsignalsignalAll 操作,包括带有超时的,以及一些检测、监控的方法。

通过改变一些设计决策,ConditionObject 类有效地将条件(conditions)与其它同步操作结合到了一起。该类只支持 Java 风格的管程访问规则,这些规则中,仅当当前线程持有锁且要操作的条件(condition)属于该锁时,条件操作才是合法的(一些替代操作的讨论参考4)。这样,一个 ConditionObject 关联到一个 ReentrantLock 上就表现的跟内置的管程(通过 Object.wait 等)一样了。两者的不同仅仅在于方法的名称、额外的功能以及用户可以为每个锁声明多个条件。

ConditionObject 使用了与同步器一样的内部队列节点。但是,是在一个单独的条件队列中维护这些节点的。signal 操作是通过将节点从条件队列转移到锁队列中来实现的,而没有必要在需要唤醒的线程重新获取到锁之前将其唤醒。

基本的 await 操作如下:

create and add new node to conditon queue;
 
release lock;
 
block until node is on lock queue;
 
re-acquire lock;

signal 操作如下:

transfer the first node from condition queue to lock queue;

因为只有在持有锁的时候才能执行这些操作,因此他们可以使用顺序链表队列操作来维护条件队列(在节点中用一个 nextWaiter 字段)。转移操作仅仅把第一个节点从条件队列中的链接解除,然后通过 CLH 插入操作将其插入到锁队列上。

实现这些操作主要复杂在,因超时或 Thread.interrupt 导致取消了条件等待时,该如何处理。“取消”和“唤醒”几乎同时发生就会有竞态问题,最终的结果遵照内置管程相关的规范。JSR133 修订以后,就要求如果中断发生在 signal 操作之前,await 方法必须在重新获取到锁后,抛出 InterruptedException。但是,如果中断发生在 signal 后,await 必须返回且不抛异常,同时设置线程的中断状态。

为了维护适当的顺序,队列节点状态变量中的一个位记录了该节点是否已经(或正在)被转移。“唤醒”和“取消”相关的代码都会尝试用 compareAndSet 修改这个状态。如果某次 signal 操作修改失败,就会转移队列中的下一个节点(如果存在的话)。如果某次“取消”操作修改失败,就必须中止此次转移,然后等待重新获得锁。后面的情况采用了一个潜在的无限的自旋等待。在节点成功的被插到锁队列之前,被“取消”的等待不能重新获得锁,所以必须自旋等待 CLH 队列插入(即 compareAndSet 操作)被“唤醒”线程成功执行。这里极少需要自旋,且自旋里使用 Thread.yield 来提示应该调度某一其它线程,理想情况下就是执行 signal 的那个线程。虽然有可能在这里为“取消”实现一个帮助策略以帮助插入节点,但这种情况实在太少,找不到合适的理由来增加这些开销。在其它所有的情况下,这个基本的机制都不需要自旋或 yield,因此在单处理器上保持着合理的性能。

4 用法

AQS 类将上述的功能结合到一起,并且作为一种基于“模版方法模式”6的基类提供给同步器。子类只需定义状态的检查与更新相关的方法,这些方法控制着 acquirerelease 操作。然而,将 AQS 的子类作为同步器 ADT 并不适合,因为这个类必须提供方法在内部控制 acquirerelease 的规则,这些都不应该被用户所看到。所有 java.util.concurrent 包中的同步器类都声明了一个私有的继承了 AbstractQueuedSynchronizer 的内部类,并且把所有同步方法都委托给这个内部类。这样各个同步器类的公开方法就可以使用适合自己的名称。

下面是一个最简单的 Mutex 类的实现,它使用同步状态 0 表示解锁,1 表示锁定。这个类并不需要同步方法中的参数,因此这里在调用的时候使用 0 作为实参,方法实现里将其忽略。

class Mutex {
    class Sync extends AbstractQueuedSynchronizer {
        public boolean tryAcquire(int ignore) {
            return compareAndSetState(0, 1);
        }
 
        public boolean tryRelease(int ignore) {
            setState(0); return true;
        }
    }
 
    private final Sync sync = new Sync();
    public void lock() { sync.acquire(0); }
    public void unlock() { sync.release(0); }
}

这个例子的一个更完整的版本,以及其它用法指南,可以在 J2SE 的文档中找到。还可以有一些变体。如,tryAcquire 可以使用一种“test-and-test-and-set”策略,即在改变状态值前先对状态进行校验。

令人诧异的是,像互斥锁这样性能敏感的东西也打算通过委托和虚方法结合的方式来定义。然而,这正是现代动态编译器一直在重点研究的面向对象设计结构。编译器擅长将这方面的开销优化掉,起码会优化频繁调用同步器的那些代码。

AbstractQueuedSynchronizer 类也提供了一些方法用来协助策略控制。例如,基础的 acquire 方法有可超时和可中断的版本。虽然到目前为止,我们的讨论都集中在像锁这样的独占模式的同步器上,但 AbstractQueuedSynchronizer 类也包含另一组方法(如 acquireShared),它们的不同点在于 tryAcquireSharedtryReleaseShared 方法能够告知框架(通过它们的返回值)尚能接受更多的请求,最终框架会通过级联的 signal(cascading signals) 唤醒多个线程。

虽然将同步器序列化(持久化存储或传输)一般来说没有太大意义,但这些类经常会被用于构造其它类,例如线程安全的集合,而这些集合通常是可序列化的。AbstractQueuedSynchronizerConditionObject 类都提供了方法用于序列化同步状态,但不会序列化潜在的被阻塞的线程,也不会序列化其它内部暂时性的簿记(bookkeeping)变量。即使如此,在反序列化时,大部分同步器类也只仅将同步状态重置为初始值,这与内置锁的隐式策略一致 —— 总是反序列化到一个解锁状态。这相当于一个空操作,但仍必须显式地支持以便 final 字段能够反序列化。

4.1 公平调度的控制

尽管同步器是基于 FIFO 队列的,但它们并不一定就得是公平的。可以注意到,在基础的 acquire 算法(3.3 节)中,tryAcquire 是在入队前被执行的。因此一个新的 acquire 线程能够“窃取”本该属于队列头部第一个线程通过同步器的机会。

可闯入的 FIFO 策略通常会提供比其它技术更高的总吞吐率。当一个有竞争的锁已经空闲,而下一个准备获取锁的线程又正在解除阻塞的过程中,这时就没有线程可以获取到这个锁,如果使用闯入策略,则可减少这之间的时间间隔。与此同时,这种策略还可避免过分的,无效率的竞争,这种竞争是由于只允许一个(第一个)排队的线程被唤醒然后尝试 acquire 操作导致的。在只要求短时间持有同步器的场景中,创建同步器的开发者可以通过定义 tryAcquire 在控制权返回之前重复调用自己若干次,来进一步凸显闯入的效果。

可闯入的 FIFO 同步器只有概率性的公平属性。锁队列头部一个解除了阻塞的线程拥有一次无偏向的机会(译者注:即不会偏向队头的线程也不会偏向闯入的线程)来赢得与闯入的线程之间的竞争,如果竞争失败,要么重新阻塞要么进行重试。然而,如果闯入的线程到达的速度比队头的线程解除阻塞快,那么在队列中的第一个线程将很难赢得竞争,以至于几乎总要重新阻塞,并且它的后继节点也会一直保持阻塞。对于短暂持有的同步器来说,在队列中第一个线程被解除阻塞期间,多处理器上很可能发生过多次闯入(译者注:即闯入的线程的 acquire 操作)和 release 了。正如下文所提到的,最终结果就是保持一或多个线程的高进展速度的同时,仍至少在一定概率上避免了饥饿的发生。

当有更高的公平性需求时,实现起来也很简单。如果需要严格的公平性,程序员可以把 tryAcquire 方法定义为,若当前线程不是队列的头节点(可通过 getFirstQueuedThread 方法检查,这是框架提供的为数不多的几个检测方法之一),则立即失败(返回 false)。

一个更快,但非严格公平的变体可以这样做,若队列为空(判断的瞬间),仍然允许 tryAcquire 执行成功。在这种情况下,多个线程同时遇到一个空队列时可能会去竞争以使自己第一个获得锁,这样,通常至少有一个线程是无需入队列的。java.util.concurrent 包中所有支持公平模式的同步器都采用了这种策略。

尽管公平性设置在实践中很有用,但是它们并没有保障,因为 Java Language Specification 没有提供这样的调度保证。例如:即使是严格公平的同步器,如果一组线程不需要同步(注:即它们具有完全并行性,执行结果无关各自的执行顺序),那么 JVM 可能会决定纯粹以顺序方式运行它们。在实际中,单处理器上,在抢占式上下文切换之前,这样的线程有可能是各自运行了一段时间。如果这样一个线程正持有某个互斥锁,它将很快会被切换回来,仅是为了释放其持有的锁,然后会继续阻塞,因为它知道有另外一个线程需要这把锁,这更增加了同步器可用但没有线程能来获取之间的间隔。同步器公平性设置在多处理器上的影响可能会更大,因为在这种环境下会产生更多的交错,因此一个线程就会有更多的机会发现锁被另一个线程请求。

在高竞争下,当保护的是短暂持有锁的代码体时,公平锁性能可能会较差。但如果,公平性锁保护的是有相对长的临界区或有相对长的锁间(inter-lock)间隔的代码,它将相对非公平锁更优,因为在这种情况下,闯入只能带来很小的性能优势,但却可能大大增加无限等待的风险。同步器框架将这些工程决策留给用户来确定。

4.2 同步器

下面是 java.util.concurrent 包中同步器定义方式的概述:

ReentrantLock 类使用 AQS 同步状态来保存锁(重复)持有的次数。当锁被一个线程获取时,ReentrantLock 也会记录下当前获得锁的线程标识,以便检查是否是重复获取,以及当错误的线程(译者注:如果线程不是锁的持有者,在此线程中执行该锁的 unlock 操作就是非法的)试图进行解锁操作时检测是否存在非法状态异常。ReentrantLock 也使用了 AQS 提供的 ConditionObject,还向外暴露了其它监控和监测相关的方法。ReentrantLock 通过在内部声明两个不同的 AbstractQueuedSynchronizer 实现类(提供公平模式的那个禁用了闯入策略)来实现可选的公平模式,在创建 ReentrantLock 实例的时候根据设置(译者注:即 ReentrantLock 构造方法中的 fair 参数)使用相应的 AbstractQueuedSynchronizer 实现类。

ReentrantReadWriteLock 类使用 AQS 同步状态中的 16位 来保存写锁持有的次数,剩下的 16位 用来保存读锁的持有次数。WriteLock 的构建方式同 ReentrantLockReadLock 则通过使用 acquireShared 方法来支持同时允许多个读线程。

Semaphore 类(计数信号量)使用 AQS 同步状态来保存信号量的当前计数。它里面定义的 acquireShared 方法会减少计数,或当计数为非正值时阻塞线程;tryRelease 方法会增加计数,可能在计数为正值时还要解除线程的阻塞。

CountDownLatch 类使用 AQS 同步状态来表示计数。当该计数为 0 时,所有的 acquire 操作(译者注:acquire 操作是从 aqs 的角度说的,对应到 CountDownLatch 中就是 await 方法)才能通过。

FutureTask 类使用 AQS 同步状态来表示某个异步计算任务的运行状态(初始化、运行中、被取消和完成)。设置(译者注:FutureTaskset 方法)或取消(译者注:FutureTaskcancel 方法)一个 FutureTask 时会调用 AQSrelease 操作,等待计算结果的线程的阻塞解除是通过 AQSacquire 操作实现的。

SynchronousQueues 类(一种 CSP(Communicating Sequential Processes) 形式的传递)使用了内部的等待节点,这些节点可以用于协调生产者和消费者。同时,它使用 AQS 同步状态来控制当某个消费者消费当前一项时,允许一个生产者继续生产,反之亦然。

java.util.concurrent 包的使用者当然也可以为自定义的应用定义自己的同步器。例如,那些曾考虑到过的,但没有采纳进这个包的同步器包括提供 WIN32 事件各种风格的语义类,二元信号量,集中管理的锁以及基于树的屏障。

5 性能

虽然 AQS 框架除了支持互斥锁外,还支持其它形式的同步方式,但锁的性能是最容易测量和比较的。即使如此,也还存在许多不同的测量方式。这里的实验主要是设计来展示锁的开销和吞吐量。

在每个测试中,所有线程都重复的更新一个伪随机数,该随机数由 nextRandom(int seed) 方法计算:

int t = (seed % 127773) * 16807 - (seed / 127773) * 2836;
 
return (t > 0) ? t : t + 0x7fffffff;

在每次迭代中,线程以概率 S 在一个互斥锁下更新共享的生成器,否则(译者注:概率为 1-S)更新其自己局部的生成器,此时是不需要锁的。如此,锁占用区域的耗时是短暂的,这就使线程持有锁期间被抢占时的外界干扰降到了最小。这个函数的随机性主要是为了两个目的:确定是否需要使用锁(这个生成器足以应付这里的需求),以及使循环中的代码不可能被轻易地优化掉。

这里比较了四种锁:内置锁,用的是 synchronized 块;互斥锁,用的是像第四节例子中的那样简单的 Mutex 类;可重入锁,用的是 ReentrantLock;以及公平锁,用的是 ReentrantLock 的公平模式。所有测试都运行在 J2SE1.5 JDK build46(大致与 beta2 相同)的 server 模式下。在收集测试数据前,测试程序先运行 20 次非竞争的测试,以排除 JVM“预热” 过程的影响。除了公平模式下的测试只跑了一百万次迭代,其它每个线程中的测试都运行了一千万次迭代。

该测试运行在四个 X86 机器和四个 UltraSparc 机器上。所有 X86 机器都运行的是 RedHat 基于 NPTL 2.4 内核和库的 Linux 系统。所有的 UltraSparc 机器都运行的是 Solaris-9。测试时所有系统的负载都很轻。根据该测试的特征,并不要求系统完全空闲(译者注:即测试时操作系统上有其它较轻的负载也不会影响本次测试的结果。)。“4P”这个名字反映出双核超线程的 Xeon 更像是 4 路机器,而不是 2 路机器。这里没有将测试数据规范化。如下所示,同步的相对开销与处理器的数量、类型、速度之间不具备简单的关系。

表1 测试的平台

名字 处理器数量 类型 速度(Mhz)
1P 1 Pentium3 900
2P 2 Pentium3 1400
2A 2 Athlon 2000
4P 2HT Pentium4/Xeon 2400
1U 1 UltraSparc2 650
4U 4 UltraSparc2 450
8U 8 UltraSparc3 750
24U 24 UltraSparc3 750

5.1 开销

无竞争情况下的开销是通过仅运行一个线程,将概率 S 为 1 时的每次迭代时间减去概率 S 为 0(访问共享内存的概率为 0)时的每次迭代时间得到的(译者注:这里的“概率 S”即前文提到的“概率 S”,概率为 0 时是没有锁操作的,概率为 1 时是每次都有锁操作,因此将概率为 1 时的耗时减去概率为 0 时的耗时就是整个锁操作的开销。)。表2 以纳秒为单位显示了非竞争场景下每次锁操作的开销。Mutex 类最接近于框架的基本耗时,可重入锁的额外开销是记录当前所有者线程和错误检查的耗时,对于公平锁来说还包含开始时检查队列是否为空的耗时。

表格 2 也展示与内置锁的“快速路径(fast path)”对比,tryAcquire 的耗时。这里的差异主要反映出了各锁和机器中使用的不同的原子指令以及内存屏障的耗时。在多处理器上,这些指令常常是完全优于所有其它指令的。内置锁和同步器类之间的主要差别,显然是由于 Hotspot 锁在锁定和解锁时都使用了一次 compareAndSet,而同步器的 acquire 操作使用了一次 compareAndSet,但 release 操作用的是一次 volatile 写(即,多处理器上的一次内存屏障以及所有处理器上的重排序限制)。每个锁的绝对的和相对耗时因机器的不同而不同。

表2 无竞争时的单锁开销(单位:纳秒)

机器 内置 互斥 可重入 公平可重入
1P 18 9 31 37
2P 58 71 77 81
2A 13 21 31 30
4P 116 95 109 117
1U 90 40 58 67
4U 122 82 100 115
8U 160 83 103 123
24U 161 84 108 119

从另一个极端看,表3 展示了概率 S 为 1,运行 256 个并发线程时产生了大规模的锁竞争下每个锁的开销。在完全饱和的情况下,可闯入的 FIFO 锁比内置锁的开销少了一个数量级(也就是更大的吞吐量),比公平锁更是少了两个数量级。这反映出有着极大的竞争时,可闯入 FIFO 策略在维持线程进展方面的效率。

表3 也说明了即使在内部开销比较低的情况下,公平锁的性能也完全是由上下文切换的时间所决定的。列出的时间大致上都与各平台上线程阻塞和解除线程阻塞的时间相称。

表3 饱和时的单锁开销(单位:纳秒)

机器 内置 互斥 可重入 公平可重入
1P 521 46 67 8327
2P 930 108 132 14967
2A 748 79 84 33910
4P 1146 188 247 15328
1U 879 153 177 41394
4U 2590 347 368 30004
8U 1274 157 174 31084
24U 1983 160 182 32291

此外,后面增加的一个实验(仅使用机器 4P)表明,对于短暂持有的锁,公平参数的设置对各线程性能方差的影响不大(注:即证明在锁被短暂持有时,即使没有公平锁,线程调度也还算公平)。这里将线程终止时间间的差异记录成一个粗粒度的离散量数。在 4P 的机器上,公平锁的时间度量的标准差平均为 0.7%,可重入锁平均为 6.0%。作为对比,为模拟一个长时间持有锁的场景,测试中使每个线程在持有锁的情况下计算了 16K 次随机数。这时,总运行时间几乎是相同的(公平锁:9.79s,可重入锁:9.72s)。公平模式下的性能方差依然很小,标准差平均为 0.1%,而可重入锁的上升到了平均 29.5%。 (注:即证明在锁被长时间持有时,使用公平锁基本不增加总时间开销,却能极大的保证公平性

5.2 吞吐量

大部分同步器都是用于无竞争和极大竞争之间的情形的。这可以用实验在两个方面进行检查,通过修改固定个线程的竞争概率,或通过往拥有固定竞争概率的线程集合里增加更多的线程。为了说明这些影响,测试运行在不同的竞争概率和不同的线程数目下,都用的是可重入锁。附图使用了一个 slowdown 度量标准。

throughout

这里,t 是总运行时间,b 是一个线程在没有竞争或同步下的基线时间,n 是线程数,p 是处理器数,S 是共享访问的比例(译者注:即前面的竞争概率 S)。计算结果是实际执行时间与理想执行时间(通常是无法得到的)的比率,理想执行时间是通过使用 Amdahl’s 法则计算出来的。理想时间模拟了一次没有同步开销,没有因锁争用而导致线程阻塞的执行过程。即使这样,在很低的竞争下,相比理想时间,有一些测试结果却表现出了很小的速度增长,大概是由于基线和测试之间的优化、流水线等方面有着轻微的差别。

graph

图中用以 2 为底的对数为比例进行了缩放。例如,值为 1 表示实际时间是理想时间的两倍,4 表示慢 16 倍。使用对数就不需要依赖一个随意的基线时间(这里指的是计算随机数的时间),因此,基于不同底数计算的结果表现出的趋势应该是类似的。这些测试使用的竞争概率从 1/128(标识为“0.008”)到1,以 2 的幂为步长,线程的数量从 1 到 1024,以 2 的幂的一半为步长。

在单处理器(1P 和 1U)上,性能随着竞争的上升而下降,但不会随着线程数的增加而下降。多处理器在遭遇竞争时,性能下降的更快。根据多处理器相关的图表显示,开始出现的峰值处虽然只有几个线程的竞争,但相对性能通常却最差。这反映出了一个性能的过渡区域,在这里闯入的线程和被唤醒的线程都准备获取锁,这会让它们频繁的迫使对方阻塞。在大部分时候,过渡区域后面会紧接着一个平滑区域,因为此时几乎没有空闲的锁,所以会与单处理器上顺序执行的模式差不多;在多处理器机器上会较早进入平滑区域。例如,请注意,在满竞争(标识为“1.000”)下这些值表示,在处理器越少的机器上,会有更糟糕的相对速度下降。

根据这些结果,可以针对阻塞(park/unpark)做进一步调优以减少上下文切换和相关的开销,这会给本框架带来小但显著的提升。此外,在多处理器上为短时间持有的但高竞争的锁采用某种形式的适应性自旋,可以避免这里看到的一些波动,这对同步器类大有裨益。虽然在跨不同上下文时适应性自旋很难很好的工作,但可以使用本框架为遇到这类使用配置的特定应用构建一个自定义形式的锁。

6 总结

本文撰写之时,java.util.concurrent 包中的同步器框架还太新所以还不能在实践中使用。因此在 J2SE 1.5 最终版本发布之前都很难看到其大范围的使用,并且,它的设计,API 实现以及性能肯定还有无法预料的后果。但是,此时,这个框架明显能胜任其基本的目标,即为创建新的同步器提供一个高效的基础。

7 致谢

感谢这个框架的开发过程中 Dave Dice 无数的想法和建议,Mark MoirMichael Scott 使用 CLH 队列的考虑,David Holmes 对早期版本的代码和 API 的批评,Victor LuchangcoBill Scherer 修订以前版本的源代码,以及 JSR166 专家小组的其他成员(Joe Bowbeer, Josh Bloch, Brian Goetz, David Holmes, Tim Peierls)和 Bill Pugh 帮助设计和规范以及对本文草稿进行评论。这项工作的一部分是基于 DARPA PCES 的资助、NSF 资助的 EIA -0080206(用于访问 24 路 Sparc)和 Sun 的合作研究资助完成的。

参考文献:

[1] Agesen, O., D. Detlefs, A. Garthwaite, R. Knippel, Y. S.Ramakrishna, and D. White. An Efficient Meta-lock for Implementing Ubiquitous Synchronization. ACM OOPSLA Proceedings, 1999.

[2] Andrews, G. Concurrent Programming. Wiley, 1991.

[3] Bacon, D. Thin Locks: Featherweight Synchronization for Java. ACM PLDI Proceedings, 1998.

[4] Buhr, P. M. Fortier, and M. Coffin. Monitor Classification,ACM Computing Surveys, March 1995.

[5] Craig, T. S. Building FIFO and priority-queueing spin locks from atomic swap. Technical Report TR 93-02-02,Department of Computer Science, University of Washington, Feb. 1993.

[6] Gamma, E., R. Helm, R. Johnson, and J. Vlissides. Design Patterns, Addison Wesley, 1996.

[7] Holmes, D. Synchronisation Rings, PhD Thesis, Macquarie University, 1999.

[8] Magnussen, P., A. Landin, and E. Hagersten. Queue locks on cache coherent multiprocessors. 8th Intl. Parallel Processing Symposium, Cancun, Mexico, Apr. 1994.

[9] Mellor-Crummey, J.M., and M. L. Scott. Algorithms for Scalable Synchronization on Shared-Memory Multiprocessors. ACM Trans. on Computer Systems,February 1991

[10] M. L. Scott and W N. Scherer III. Scalable Queue-Based Spin Locks with Timeout. 8th ACM Symp. on Principles and Practice of Parallel Programming, Snowbird, UT, June 2001.

[11] Sun Microsystems. Multithreading in the Solaris Operating Environment. White paper available at http://wwws.sun.com/software/solaris/whitepapers.html 2002.

[12] Zhang, H., S. Liang, and L. Bak. Monitor Conversion in a Multithreaded Computer System. United States Patent 6,691,304. 2004.


推荐阅读:

从ReentrantLock的实现看AQS的原理及应用

最后附上 GitHub:<https://github.com/gonearew