AQS(4):condition

作者: Cafebaby | 来源:发表于2016-04-03 15:15 被阅读114次

AQS中提供了一个实现Condition接口的内部类ConditionObject,其内部也维护了一个队列,首尾分别为firstWaiter,lastWaiter。当然,condition中最重要的两类方法就是await和signal。

首先,我们来看await方法,在接口注释中我们知道该方法会导致当前线程等待直到被唤醒或者被中断。与该condition关联的锁会被自动释放并阻塞当前线程。在这个方法返回之前必须重新拥有锁。这里,我们先来看awaitUninterruptibly方法,这个方法是不响应中断的。

public final void awaitUninterruptibly() {     
    Node node = addConditionWaiter();    
    int savedState = fullyRelease(node);   
    boolean interrupted = false;    
    while (!isOnSyncQueue(node)) {  
        LockSupport.park(this); 
        if (Thread.interrupted())    
            interrupted = true;    
     }   
     if (acquireQueued(node, savedState) ||   interrupted)        
        selfInterrupt();
  }

该方法新建了一个Node链接到lastWaiter之后。然后调用fullyRelease方法,该方法内部调用了release方法解锁了同步队列中头结点的一个后继节点,可参考AQS(3)。然后判断当前节点是否在同步队列中,如果不在,那么阻塞当前线程。直到while条件为true,此时当前线程处于等待队列,并且需要再次acquire。

然后我们来看signal方法,先贴代码

private void doSignal(Node first) {    
    do {        
        if ( (firstWaiter = first.nextWaiter) == null)            
            lastWaiter = null;        
        first.nextWaiter = null;    
    } 
    while (!transferForSignal(first) && (first = firstWaiter) != null);
}

这个方法从头节点开始,然后调用了transferForSignal方法,将头结点从条件队列移到了等待队列。此时会将ws用CAS置为0。然后将节点用enq方法入队并返回其前驱结点,需要将前驱节点用CAS设置为SIGNAL表示其后继需要被解锁。

我们将两个方法结合起来看,首先一个线程调用await,那么会新建一个节点加入条件队列,然后它会释放持有的锁。这时它仍然在条件队列,因此在第一次循环中会被阻塞。while循环保证了中断或者假唤醒不会导致出错。这时,另一线程调用了signal操作,它从条件队列拿出第一个节点,把它加入了同步队列,并将其前驱标记为SINGNAL指示需要解锁该节点。(这个时候仍然没有被唤醒),我们知道在signal的时候仍然需要持有锁,当这个线程在释放锁的时候,会在release方法中解锁一个后继节点,此时同步队列只有这个被阻塞的线程,这时他就被唤醒了!

除了上述方法外,还有signalAll方法,但该方法的区别就是将所有条件队列的节点都加到了同步队列。

而在await系列方法中,await()方法是支持中断的。该方法会判断中断的时机,如果在signalled之前中断则抛出异常,如果在signalled之后中断则reinterrupt。其他诸如awaitNano,awaitUntil等在主体逻辑上并没有区别。

到这里,AQS这部分的简单介绍就结束啦!

相关文章

  • 知识梳理目录

    Java基础 Java线程池 AQS之独占锁 AQS之Condition AQS之Condition AQS之同步...

  • AQS(4):condition

    AQS中提供了一个实现Condition接口的内部类ConditionObject,其内部也维护了一个队列,首尾分...

  • AQS之Condition

    AQS中还实现了Condition的功能。本文将通过ReentrantLock来分析在AQS中Condition的...

  • JUC-(11)AQS(中)-共享模式

    AQS(上)-独占模式 AQS(中)-共享模式 AQS(下)-Condition 对比 独占模式共享模式acqui...

  • AQS的ConditionObject源码详解

    Condition接口见Condition接口详解 ConditionObject定义 定义为AQS的public...

  • JUC-(10)AQS(上)-独占模式

    AQS(上)-独占模式 AQS(中)-共享模式 AQS(下)-Condition 简叙 之前我写过很多关于JUC下...

  • JUC-(12)AQS(下)-Condition

    AQS(上)-独占模式 AQS(中)-共享模式 AQS(下)-Condition在之前的文章中已经介绍了独占模式和...

  • J.U.C:Condition

    Condition源码分析   调用 Condition,需要获得 Lock 锁,所以意味着会存在一个 AQS 同...

  • AQS之同步器

    AQS之独占锁AQS之共享锁AQS之Condition 在了解了AQS的实现原理之后再来看这些同步器会觉得很亲切,...

  • AQS之Condition

    使用示例 调用Condition#await方法会释放当前持有的锁,然后阻塞当前线程,同时向Condition队列...

网友评论

    本文标题:AQS(4):condition

    本文链接:https://www.haomeiwen.com/subject/xqdulttx.html