@ ThreadSafe public abstract class BaseBoundedBuffer<E> { @GuardeBy( "this" ) private final E[] buf; @GuardeBy( "this" ) private int tail; @GuardeBy( "this" ) private int head; @GuardeBy( "this" ) private int count; protected BaseBoundedBuffer( int capacity) { this .buf = (E[]) new Object[capacity]; } protected synchronized final void doPut(E E) { buf[tail] = E; if (++tail == buf.length) { tail = 0; } ++count; } protected synchronized final E doTake() { E E = buf[head]; buf[head] = null ; if (++head == buf.length) { head = 0; } --count; return E; } public synchronized final boolean isFull() { return count == buf.length; } public synchronized final boolean isEmpty() { return count == 0; } }
@ ThreadSafe public class GrumpyBoundedBuffer<V> extends BaseBoundedBuffer<V> { public GrumpyBoundedBuffer( int size){ super (size); } public synchronized void put(V v){ if (isFull()){ throw new BufferFullException (); } doPut(v); } public synchronized V take(){ if (isEmpty()) throw new BufferEmptyExeption (); return doTake(); } }
while (true ){ try { V item = buffer.take(); // 對於item運行一些操做 break ; } catch (BufferEmptyException e) { Thread. sleep(SLEEP_GRANULARITY ); } }
@ ThreadSafe public class SleepyBoundedBuffer<V> extends BaseBoundedBuffer<V> { public SleepyBoundedBuffer( int size) { super (size); } public void put(V v) throws InterruptedException{ while (true ){ synchronized (this ){ if (!isFull()){ doPut(v); return ; } } Thread.sleep(SLEEP_GRANULARITY); } } public V take() throws InterruptedException{ while (true ){ synchronized (this ){ if (!isEmpty()){ return doTake(); } } Thread.sleep(SLEEP_GRANULARITY); } } }
@ ThreadSafe public class BoundedBuffer<V> extends BaseBoundedBuffer<V> { // 條件謂詞:not-full (!isFull()) // 條件謂詞:not-empty (!isEmpty()) public BoundedBuffer( int size) { super (size); } // 堵塞並直道:not-full public synchronized void put(V v) throws InterruptedException{ while (isFull()){ wait(); } doPut(v); notifyAll(); } // 堵塞並直道:not-empty public synchronized V take() throws InterruptedException{ while (isEmpty()){ wait(); } V v = doTake(); notifyAll(); return v; } }
. 在一個循環中調用wait。
. 確保使用與條件隊列相關的鎖來保護構成條件謂詞的各個狀態變量。
. 當調用wait、notify或notifyAll等方法時。必定要持有與條件隊列相關的鎖。
. 在檢查條件謂詞以後以及開始運行對應的操做以前,不要釋放鎖。java