Java中的condition类
Condition是在java 1.5中才呈现的,它用来代替传统的Object的wait()、notify()实现线程间的合作,
相比应用Object的wait()、notify(),应用Condition的await()、signal()这种形式实现线程间合作更加平安和高效。
因而通常来说,线程通信的实现比拟举荐应用Condition
- Condition是个接口,根本的办法就是await()和signal()办法;
- Condition依赖于Lock接口,生成一个Condition的根本代码是lock.newCondition()
- 调用Condition的await()和signal()办法,都必须在lock爱护之内,就是说必须在lock.lock()和lock.unlock之间才能够应用,因为外部会做开释锁的操作,如果不是在lock和unlock之间应用,会报错
java.lang.IllegalMonitorStateException
Conditon中的await()对应Object的wait();
Condition中的signal()对应Object的notify();
Condition中的signalAll()对应Object的notifyAll()。
简略demo
Lock lock = new ReentrantLock(); Condition condition = lock.newCondition(); public void conditionWait() throws InterruptedException { lock.lock(); try { condition.await(); } finally { lock.unlock(); } } public void conditionSignal() throws InterruptedException { lock.lock(); try { condition.signal(); } finally { lock.unlock(); } }
个别都会将Condition对象作为成员变量。当调用await()办法后,以后线程会开释锁并在此期待,而其余线程调用Condition对象的signal()办法,告诉以后线程后,以后线程才从await()办法返回,并且在返回前曾经获取了锁。
深刻了解demo
获取一个Condition必须通过Lock的newCondition()办法。上面通过一个有界队列的示例来深刻理解Condition的应用形式。有界队列是一种非凡的队列,当队列为空时,队列的获取操作将会阻塞获取线程,直到队列中有新增元素,当队列已满时,队列的插入操作将会阻塞插入线程,直到队列呈现“空位”
public class BoundedQueue<T> { private Object[] items; // 增加的下标,删除的下标和数组以后数量 private int addIndex,removeIndex,count; private Lock lock = new ReentrantLock(); private Condition notEmpty = lock.newCondition(); private Condition notFull = lock.newCondition(); public BoundedQueue(int size){ items = new Object[size]; } /** * 增加一个元素,如果数组满,则增加线程进入期待状态,直到有"空位" * @author fuyuwei * 2017年5月21日 下午6:14:55 * @param t * @throws InterruptedException */ public void add(T t) throws InterruptedException{ lock.lock(); try{ while(count == items.length){ notFull.await(); } items[addIndex] = t; if(++addIndex == items.length) addIndex = 0; ++count; notEmpty.signal(); }finally{ lock.unlock(); } } /** * 由头部删除一个元素,如果数组空,则删除线程进入期待状态,直到有新增加元素 * @author fuyuwei * 2017年5月21日 下午6:20:54 * @return * @throws InterruptedException */ @SuppressWarnings("unchecked") public T remove() throws InterruptedException{ lock.lock(); try{ while(count == 0) notEmpty.await(); Object x = items[removeIndex]; if(++removeIndex == items.length) removeIndex = 0; --count; notFull.signal(); return (T)x; }finally{ lock.unlock(); } }}
首先须要取得锁,目标是确保数组批改的可见性和排他性。当数组数量等于数组长度时,示意数组已满,则调用notFull.await(),以后线程随之开释锁并进入期待状态。如果数组数量不等于数组长度,示意数组未满,则增加元素到数组中,同时告诉期待在notEmpty上的线程,数组中曾经有新元素能够获取。在增加和删除办法中应用while循环而非if判断,目标是避免过早或意外的告诉,只有条件合乎才可能退出循环。
原理
期待队列
期待队列是一个FIFO的队列,在队列中的每个节点都蕴含了一个线程援用,该线程就是在Condition对象上期待的线程,如果一个线程调用了Condition.await()办法,那么该线程将会开释锁、结构成节点退出期待队列并进入期待状态
一个Condition蕴含一个期待队列,Condition领有首节点(firstWaiter)和尾节点(lastWaiter)。以后线程调用Condition.await()办法,将会以以后线程结构节点,并将节点从尾部退出期待队列,期待队列的根本构造如下图所示
如图所示,Condition领有首尾节点的援用,而新增节点只须要将原有的尾节点nextWaiter指向它,并且更新尾节点即可。上述节点援用更新的过程并没有应用CAS保障,起因在于调用await()办法的线程必然是获取了锁的线程,也就是说该过程是由锁来保障线程平安的。在Object的监视器模型上,一个对象领有一个同步队列和期待队列,而并发包中的Lock(更确切地说是同步器)领有一个同步队列和多个期待队列,其对应关系如下图所示
期待
调用Condition的await()办法(或者以await结尾的办法),会使以后线程进入期待队列并开释锁,同时线程状态变为期待状态。当从await()办法返回时,以后线程肯定获取了Condition相关联的锁。如果从队列(同步队列和期待队列)的角度看await()办法,当调用await()办法时,相当于同步队列的首节点(获取了锁的节点)挪动到Condition的期待队列中
public final void await() throws InterruptedException { if (Thread.interrupted()) throw new InterruptedException(); // 以后线程退出期待队列 Node node = addConditionWaiter(); // 开释同步状态,也就是开释锁 int savedState = fullyRelease(node); int interruptMode = 0; while (!isOnSyncQueue(node)) { LockSupport.park(this); if ((interruptMode = checkInterruptWhileWaiting(node)) != 0) break; } if (acquireQueued(node, savedState) && interruptMode != THROW_IE) interruptMode = REINTERRUPT; if (node.nextWaiter != null) unlinkCancelledWaiters(); if (interruptMode != 0) reportInterruptAfterWait(interruptMode); }
调用该办法的线程胜利获取了锁的线程,也就是同步队列中的首节点,该办法会将以后线程结构成节点并退出期待队列中,而后开释同步状态,唤醒同步队列中的后继节点,而后以后线程会进入期待状态。当期待队列中的节点被唤醒,则唤醒节点的线程开始尝试获取同步状态。如果不是通过其余线程调用Condition.signal()办法唤醒,而是对期待线程进行中断,则会抛出InterruptedException
告诉
调用Condition的signal()办法,将会唤醒在期待队列中等待时间最长的节点(首节点),在唤醒节点之前,会将节点移到同步队列中
public final void signal() { if (!isHeldExclusively()) throw new IllegalMonitorStateException(); Node first = firstWaiter; if (first != null) doSignal(first); }
调用该办法的前置条件是以后线程必须获取了锁,能够看到signal()办法进行了isHeldExclusively()查看,也就是以后线程必须是获取了锁的线程。接着获取期待队列的首节点,将其挪动到同步队列并应用LockSupport唤醒节点中的线程
节点从期待队列挪动到同步队列的过程如下图所示
通过调用同步器的enq(Node node)办法,期待队列中的头节点线程平安地挪动到同步队列。当节点挪动到同步队列后,以后线程再应用LockSupport唤醒该节点的线程。被唤醒后的线程,将从await()办法中的while循环中退出(isOnSyncQueue(Node node)办法返回true,节点曾经在同步队列中),进而调用同步器的acquireQueued()办法退出到获取同步状态的竞争中。胜利获取同步状态(或者说锁)之后,被唤醒的线程将从先前调用的await()办法返回,此时该线程曾经胜利地获取了锁。
Condition的signalAll()办法,相当于对期待队列中的每个节点均执行一次signal()办法(留神是这个Condition对应的期待队列),成果就是将期待队列中所有节点全副挪动到同步队列中,并唤醒每个节点的线程。