關(guān)于并發(fā)中的阻塞

內(nèi)容摘自聊聊并發(fā)(七)——Java中的阻塞隊列

1. 什么是阻塞隊列?

阻塞隊列(BlockingQueue)是一個支持兩個附加操作的隊列。這兩個附加的操作是:在隊列為空時,獲取元素的線程會等待隊列變?yōu)榉强铡.旉犃袧M時,存儲元素的線程會等待隊列可用。阻塞隊列常用于生產(chǎn)者和消費者的場景,生產(chǎn)者是往隊列里添加元素的線程,消費者是從隊列里拿元素的線程。阻塞隊列就是生產(chǎn)者存放元素的容器,而消費者也只從容器里拿元素。

阻塞隊列提供了四種處理方法:

方法\處理方式拋出異常返回特殊值一直阻塞超時退出

插入方法add(e)offer(e)put(e)offer(e,time,unit)

移除方法remove()poll()take()poll(time,unit)

檢查方法element()peek()不可用不可用

拋出異常:是指當阻塞隊列滿時候,再往隊列里插入元素,會拋出IllegalStateException("Queue full")異常。當隊列為空時,從隊列里獲取元素時會拋出NoSuchElementException異常 。

返回特殊值:插入方法會返回是否成功,成功則返回true。移除方法,則是從隊列里拿出一個元素,如果沒有則返回null

一直阻塞:當阻塞隊列滿時,如果生產(chǎn)者線程往隊列里put元素,隊列會一直阻塞生產(chǎn)者線程,直到拿到數(shù)據(jù),或者響應中斷退出。當隊列空時,消費者線程試圖從隊列里take元素,隊列也會阻塞消費者線程,直到隊列可用。

超時退出:當阻塞隊列滿時,隊列會阻塞生產(chǎn)者線程一段時間,如果超過一定的時間,生產(chǎn)者線程就會退出。

2. Java里的阻塞隊列

JDK7提供了7個阻塞隊列。分別是

ArrayBlockingQueue :一個由數(shù)組結(jié)構(gòu)組成的有界阻塞隊列。

LinkedBlockingQueue :一個由鏈表結(jié)構(gòu)組成的有界阻塞隊列。

PriorityBlockingQueue :一個支持優(yōu)先級排序的無界阻塞隊列。

DelayQueue:一個使用優(yōu)先級隊列實現(xiàn)的無界阻塞隊列。

SynchronousQueue:一個不存儲元素的阻塞隊列。

LinkedTransferQueue:一個由鏈表結(jié)構(gòu)組成的無界阻塞隊列。

LinkedBlockingDeque:一個由鏈表結(jié)構(gòu)組成的雙向阻塞隊列。

ArrayBlockingQueue是一個用數(shù)組實現(xiàn)的有界阻塞隊列。此隊列按照先進先出(FIFO)的原則對元素進行排序。默認情況下不保證訪問者公平的訪問隊列,所謂公平訪問隊列是指阻塞的所有生產(chǎn)者線程或消費者線程,當隊列可用時,可以按照阻塞的先后順序訪問隊列,即先阻塞的生產(chǎn)者線程,可以先往隊列里插入元素,先阻塞的消費者線程,可以先從隊列里獲取元素。通常情況下為了保證公平性會降低吞吐量。我們可以使用以下代碼創(chuàng)建一個公平的阻塞隊列:

ArrayBlockingQueue fairQueue = new? ArrayBlockingQueue(1000,true);

訪問者的公平性是使用可重入鎖實現(xiàn)的,代碼如下:

public ArrayBlockingQueue(int capacity, boolean fair) {

? ? ? ? if (capacity <= 0)

? ? ? ? ? ? throw new IllegalArgumentException();

? ? ? ? this.items = new Object[capacity];

? ? ? ? lock = new ReentrantLock(fair);

? ? ? ? notEmpty = lock.newCondition();

? ? ? ? notFull =? lock.newCondition();

}

LinkedBlockingQueue是一個用鏈表實現(xiàn)的有界阻塞隊列。此隊列的默認和最大長度為Integer.MAX_VALUE。此隊列按照先進先出的原則對元素進行排序。

PriorityBlockingQueue是一個支持優(yōu)先級的無界隊列。默認情況下元素采取自然順序排列,也可以通過比較器comparator來指定元素的排序規(guī)則。元素按照升序排列。

DelayQueue是一個支持延時獲取元素的無界阻塞隊列。隊列使用PriorityQueue來實現(xiàn)。隊列中的元素必須實現(xiàn)Delayed接口,在創(chuàng)建元素時可以指定多久才能從隊列中獲取當前元素。只有在延遲期滿時才能從隊列中提取元素。我們可以將DelayQueue運用在以下應用場景:

緩存系統(tǒng)的設(shè)計:可以用DelayQueue保存緩存元素的有效期,使用一個線程循環(huán)查詢DelayQueue,一旦能從DelayQueue中獲取元素時,表示緩存有效期到了。

定時任務(wù)調(diào)度。使用DelayQueue保存當天將會執(zhí)行的任務(wù)和執(zhí)行時間,一旦從DelayQueue中獲取到任務(wù)就開始執(zhí)行,從比如TimerQueue就是使用DelayQueue實現(xiàn)的。

隊列中的Delayed必須實現(xiàn)compareTo來指定元素的順序。比如讓延時時間最長的放在隊列的末尾。實現(xiàn)代碼如下:

public int compareTo(Delayed other) {

? ? ? ? ? if (other == this) // compare zero ONLY if same object

? ? ? ? ? ? ? ? return 0;

? ? ? ? ? ? if (other instanceof ScheduledFutureTask) {

? ? ? ? ? ? ? ? ScheduledFutureTask x = (ScheduledFutureTask)other;

? ? ? ? ? ? ? ? long diff = time - x.time;

? ? ? ? ? ? ? ? if (diff < 0)

? ? ? ? ? ? ? ? ? ? return -1;

? ? ? ? ? ? ? ? else if (diff > 0)

? ? ? ? ? ? ? ? ? ? return 1;

? else if (sequenceNumber < x.sequenceNumber)

? ? ? ? ? ? ? ? ? ? return -1;

? ? ? ? ? ? ? ? else

? ? ? ? ? ? ? ? ? ? return 1;

? ? ? ? ? ? }

? ? ? ? ? ? long d = (getDelay(TimeUnit.NANOSECONDS) -

? ? ? ? ? ? ? ? ? ? ? other.getDelay(TimeUnit.NANOSECONDS));

? ? ? ? ? ? return (d == 0) ? 0 : ((d < 0) ? -1 : 1);

? ? ? ? }

如何實現(xiàn)Delayed接口

我們可以參考ScheduledThreadPoolExecutor里ScheduledFutureTask類。這個類實現(xiàn)了Delayed接口。首先:在對象創(chuàng)建的時候,使用time記錄前對象什么時候可以使用,代碼如下:

ScheduledFutureTask(Runnable r, V result, long ns, long period) {

? ? ? ? ? ? super(r, result);

? ? ? ? ? ? this.time = ns;

? ? ? ? ? ? this.period = period;

? ? ? ? ? ? this.sequenceNumber = sequencer.getAndIncrement();

}

然后使用getDelay可以查詢當前元素還需要延時多久,代碼如下:

public long getDelay(TimeUnit unit) {

? ? ? ? ? ? return unit.convert(time - now(), TimeUnit.NANOSECONDS);

? ? ? ? }

通過構(gòu)造函數(shù)可以看出延遲時間參數(shù)ns的單位是納秒,自己設(shè)計的時候最好使用納秒,因為getDelay時可以指定任意單位,一旦以納秒作為單位,而延時的時間又精確不到納秒就麻煩了。使用時請注意當time小于當前時間時,getDelay會返回負數(shù)。

如何實現(xiàn)延時隊列

延時隊列的實現(xiàn)很簡單,當消費者從隊列里獲取元素時,如果元素沒有達到延時時間,就阻塞當前線程。

long delay = first.getDelay(TimeUnit.NANOSECONDS);

? ? ? ? ? ? ? ? ? ? if (delay <= 0)

? ? ? ? ? ? ? ? ? ? ? ? return q.poll();

? ? ? ? ? ? ? ? ? ? else if (leader != null)

? ? ? ? ? ? ? ? ? ? ? ? available.await();

SynchronousQueue是一個不存儲元素的阻塞隊列。每一個put操作必須等待一個take操作,否則不能繼續(xù)添加元素。SynchronousQueue可以看成是一個傳球手,負責把生產(chǎn)者線程處理的數(shù)據(jù)直接傳遞給消費者線程。隊列本身并不存儲任何元素,非常適合于傳遞性場景,比如在一個線程中使用的數(shù)據(jù),傳遞給另外一個線程使用,SynchronousQueue的吞吐量高于LinkedBlockingQueue 和 ArrayBlockingQueue。

LinkedTransferQueue是一個由鏈表結(jié)構(gòu)組成的無界阻塞TransferQueue隊列。相對于其他阻塞隊列,LinkedTransferQueue多了tryTransfer和transfer方法。

transfer方法。如果當前有消費者正在等待接收元素(消費者使用take()方法或帶時間限制的poll()方法時),transfer方法可以把生產(chǎn)者傳入的元素立刻transfer(傳輸)給消費者。如果沒有消費者在等待接收元素,transfer方法會將元素存放在隊列的tail節(jié)點,并等到該元素被消費者消費了才返回。transfer方法的關(guān)鍵代碼如下:

Node pred = tryAppend(s, haveData);

return awaitMatch(s, pred, e, (how == TIMED), nanos);

第一行代碼是試圖把存放當前元素的s節(jié)點作為tail節(jié)點。第二行代碼是讓CPU自旋等待消費者消費元素。因為自旋會消耗CPU,所以自旋一定的次數(shù)后使用Thread.yield()方法來暫停當前正在執(zhí)行的線程,并執(zhí)行其他線程。

tryTransfer方法。則是用來試探下生產(chǎn)者傳入的元素是否能直接傳給消費者。如果沒有消費者等待接收元素,則返回false。和transfer方法的區(qū)別是tryTransfer方法無論消費者是否接收,方法立即返回。而transfer方法是必須等到消費者消費了才返回。

對于帶有時間限制的tryTransfer(E e, long timeout, TimeUnit unit)方法,則是試圖把生產(chǎn)者傳入的元素直接傳給消費者,但是如果沒有消費者消費該元素則等待指定的時間再返回,如果超時還沒消費元素,則返回false,如果在超時時間內(nèi)消費了元素,則返回true。

LinkedBlockingDeque是一個由鏈表結(jié)構(gòu)組成的雙向阻塞隊列。所謂雙向隊列指的你可以從隊列的兩端插入和移出元素。雙端隊列因為多了一個操作隊列的入口,在多線程同時入隊時,也就減少了一半的競爭。相比其他的阻塞隊列,LinkedBlockingDeque多了addFirst,addLast,offerFirst,offerLast,peekFirst,peekLast等方法,以First單詞結(jié)尾的方法,表示插入,獲取(peek)或移除雙端隊列的第一個元素。以Last單詞結(jié)尾的方法,表示插入,獲取或移除雙端隊列的最后一個元素。另外插入方法add等同于addLast,移除方法remove等效于removeFirst。但是take方法卻等同于takeFirst,不知道是不是Jdk的bug,使用時還是用帶有First和Last后綴的方法更清楚。

在初始化LinkedBlockingDeque時可以設(shè)置容量防止其過渡膨脹。另外雙向阻塞隊列可以運用在“工作竊取”模式中。

3. 阻塞隊列的實現(xiàn)原理

如果隊列是空的,消費者會一直等待,當生產(chǎn)者添加元素時候,消費者是如何知道當前隊列有元素的呢?如果讓你來設(shè)計阻塞隊列你會如何設(shè)計,讓生產(chǎn)者和消費者能夠高效率的進行通訊呢?讓我們先來看看JDK是如何實現(xiàn)的。

使用通知模式實現(xiàn)。所謂通知模式,就是當生產(chǎn)者往滿的隊列里添加元素時會阻塞住生產(chǎn)者,當消費者消費了一個隊列中的元素后,會通知生產(chǎn)者當前隊列可用。通過查看JDK源碼發(fā)現(xiàn)ArrayBlockingQueue使用了Condition來實現(xiàn),代碼如下:

private final Condition notFull;

private final Condition notEmpty;

public ArrayBlockingQueue(int capacity, boolean fair) {

? ? ? ? //省略其他代碼

? ? ? ? notEmpty = lock.newCondition();

? ? ? ? notFull =? lock.newCondition();

? ? }

public void put(E e) throws InterruptedException {

? ? ? ? checkNotNull(e);

? ? ? ? final ReentrantLock lock = this.lock;

? ? ? ? lock.lockInterruptibly();

? ? ? ? try {

? ? ? ? ? ? while (count == items.length)

? ? ? ? ? ? ? ? notFull.await();

? ? ? ? ? ? insert(e);

? ? ? ? } finally {

? ? ? ? ? ? lock.unlock();

? ? ? ? }

}

public E take() throws InterruptedException {

? ? ? ? final ReentrantLock lock = this.lock;

? ? ? ? lock.lockInterruptibly();

? ? ? ? try {

? ? ? ? ? ? while (count == 0)

? ? ? ? ? ? ? ? notEmpty.await();

? ? ? ? ? ? return extract();

? } finally {

? ? ? ? ? ? lock.unlock();

? ? ? ? }

}

private void insert(E x) {

? ? ? ? items[putIndex] = x;

? ? ? ? putIndex = inc(putIndex);

? ? ? ? ++count;

? ? ? ? notEmpty.signal();

? ? }

當我們往隊列里插入一個元素時,如果隊列不可用,阻塞生產(chǎn)者主要通過LockSupport.park(this);來實現(xiàn)

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) // clean up if cancelled

? ? ? ? ? ? ? ? unlinkCancelledWaiters();

? ? ? ? ? ? if (interruptMode != 0)

reportInterruptAfterWait(interruptMode);

? ? ? ? }

繼續(xù)進入源碼,發(fā)現(xiàn)調(diào)用setBlocker先保存下將要阻塞的線程,然后調(diào)用unsafe.park阻塞當前線程。

public static void park(Object blocker) {

? ? ? ? Thread t = Thread.currentThread();

? ? ? ? setBlocker(t, blocker);

? ? ? ? unsafe.park(false, 0L);

? ? ? ? setBlocker(t, null);

? ? }

unsafe.park是個native方法,代碼如下:

public native void park(boolean isAbsolute, long time);

park這個方法會阻塞當前線程,只有以下四種情況中的一種發(fā)生時,該方法才會返回。

與park對應的unpark執(zhí)行或已經(jīng)執(zhí)行時。注意:已經(jīng)執(zhí)行是指unpark先執(zhí)行,然后再執(zhí)行的park。

線程被中斷時。

如果參數(shù)中的time不是零,等待了指定的毫秒數(shù)時。

發(fā)生異常現(xiàn)象時。這些異常事先無法確定。

我們繼續(xù)看一下JVM是如何實現(xiàn)park方法的,park在不同的操作系統(tǒng)使用不同的方式實現(xiàn),在linux下是使用的是系統(tǒng)方法pthread_cond_wait實現(xiàn)。實現(xiàn)代碼在JVM源碼路徑src/os/linux/vm/os_linux.cpp里的 os::PlatformEvent::park方法,代碼如下:

void os::PlatformEvent::park() {? ? ?

? ? ? ? int v ;

? ? for (;;) {

v = _Event ;

? ? if (Atomic::cmpxchg (v-1, &_Event, v) == v) break ;

? ? }

? ? guarantee (v >= 0, "invariant") ;

? ? if (v == 0) {

? ? // Do this the hard way by blocking ...

? ? int status = pthread_mutex_lock(_mutex);

? ? assert_status(status == 0, status, "mutex_lock");

? ? guarantee (_nParked == 0, "invariant") ;

? ? ++ _nParked ;

? ? while (_Event < 0) {

? ? status = pthread_cond_wait(_cond, _mutex);

? ? // for some reason, under 2.7 lwp_cond_wait() may return ETIME ...

? ? // Treat this the same as if the wait was interrupted

? ? if (status == ETIME) { status = EINTR; }

? ? assert_status(status == 0 || status == EINTR, status, "cond_wait");

? ? }

? ? -- _nParked ;


? ? // In theory we could move the ST of 0 into _Event past the unlock(),

? ? // but then we'd need a MEMBAR after the ST.

? ? _Event = 0 ;

? ? status = pthread_mutex_unlock(_mutex);

? ? assert_status(status == 0, status, "mutex_unlock");

? ? }

? ? guarantee (_Event >= 0, "invariant") ;

? ? }

? ? }

pthread_cond_wait是一個多線程的條件變量函數(shù),cond是condition的縮寫,字面意思可以理解為線程在等待一個條件發(fā)生,這個條件是一個全局變量。這個方法接收兩個參數(shù),一個共享變量_cond,一個互斥量_mutex。而unpark方法在linux下是使用pthread_cond_signal實現(xiàn)的。park 在windows下則是使用WaitForSingleObject實現(xiàn)的。

當隊列滿時,生產(chǎn)者往阻塞隊列里插入一個元素,生產(chǎn)者線程會進入WAITING (parking)狀態(tài)。我們可以使用jstack dump阻塞的生產(chǎn)者線程看到這點:

"main" prio=5 tid=0x00007fc83c000000 nid=0x10164e000 waiting on condition [0x000000010164d000]

? java.lang.Thread.State: WAITING (parking)

? ? ? ? at sun.misc.Unsafe.park(Native Method)

? ? ? ? - parking to wait for? <0x0000000140559fe8> (a java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject)

? ? ? ? at java.util.concurrent.locks.LockSupport.park(LockSupport.java:186)

? ? ? ? at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await(AbstractQueuedSynchronizer.java:2043)

? ? ? ? at java.util.concurrent.ArrayBlockingQueue.put(ArrayBlockingQueue.java:324)

? ? ? ? at blockingqueue.ArrayBlockingQueueTest.main(ArrayBlockingQueueTest.java:11)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,501評論 6 544
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,673評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,610評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,939評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,668評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 56,004評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,001評論 3 449
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 43,173評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,705評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,426評論 3 359
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,656評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,139評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,833評論 3 350
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,247評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,580評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,371評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,621評論 2 380

推薦閱讀更多精彩內(nèi)容