摘要:隊列中有元素時,就說明有過期了,線程繼續執行,然后元素出隊,根據相應的移除緩存。所以嚴格來說,雖然實現了隊列接口,但是它的目的卻并不是隊列,而是將生產者消費者線程配對。轉移隊列鏈式轉移隊列。
引言
本周在編寫短信驗證碼頻率限制切面的時候,經潘老師給的實現思路,使用隊列進行實現。
看了看java.util包下的Queue接口,發現還從來沒用過呢!
Collection集合類接口,由它派生出List、Set和Queue,Map屬于另一個獨立的接口,和Collection沒有繼承關系。
List、Set和Map我們用的都是已經相當熟練了,今天,我們就來學習這個隊列Queue!
探索隊列與棧都是數據結構的基礎話題,隊列:先進先出;棧:后進先出。
方法Queue接口中聲明了六個方法,分成三對來使用。
入隊操作
方法 | 特點 | 建議 |
---|---|---|
add | 入隊失敗拋出異常 | |
offer | 入隊失敗返回false | 推薦 |
出隊操作
方法 | 特點 | 建議 |
---|---|---|
remove | 出隊失敗拋出異常 | |
poll | 出隊失敗返回null | 推薦 |
取隊頭操作
方法 | 特點 | 建議 |
---|---|---|
element | 隊列為空時拋出異常 | |
peek | 隊列為空時返回null | 推薦 |
在java.util包中,除抽象類外,直接實現Queue接口的只有PriorityQueue優先級隊列。
優先級隊列比普通的隊列要高級,普通的隊列如果是先進的肯定是在隊頭的,而優先級隊列根據優先級判斷當前隊頭元素是什么。很適合實現操作系統中的按優先級實現進程調度。
如果需要使用優先級隊列進行排序時,需要傳入比較器。
該隊列使用數組實現,線程不安全。
Dequejava.util包中,Deque接口繼承Queue接口。
Deque:double-ended queue,雙端隊列。
雙端隊列,相比普通隊列就是可操作兩端,有兩個隊頭,也有兩個隊尾。
所以再去看Deque接口中聲明的方法,都是兩套的。offerFirst、offerLast、pollFirst、pollLast等。
所以說,如果使用雙端隊列,不僅可以當隊列用,也可以當棧用,因為可以自己控制出的是隊頭還是隊尾。
Deque有兩個實現類:ArrayDeque和LinkedList。
原來LinkedList不僅實現了List接口,還實現了Deque接口。
兩者的區別顯而易見,一個是數組方式實現的,一個是鏈表的方式實現的。
BlockingQueue這些都是java.util包下的,都是線程不安全的實現,JDK所有線程安全的隊列實現都在java.util.concurrent包下,也就是阻塞隊列BlockingQueue。
在concurrent包下,自然是做了線程安全處理的了,在多線程環境下操作隊列需要使用。
生產者消費者與阻塞隊列最密切的就是生產者消費者模型了,我們一起來探討一下。
生產者消費者模型,最初出現在操作系統中,多進程/多線程進行協作,完成同一任務,必然需要相互合作與相互制約。
舉一個符合實際的例子,我想喝可樂。
可口可樂公司就是生產者,用于生產商品。
超市就相當于緩沖區,用于存儲生產者生產出來的可樂,公司生產出可樂,然后放到超市里賣。
我就是消費者,去超市買可樂(消費過程)。
所以就會有一個同步的問題:
假設場景:超市能容量100瓶可樂。
所以,消費者去購買的前提是:超市內有可樂,要不去了也買不著。
生產者生產的前提是:超市內有空余位置,要不生產了往哪送呢?
類比到程序設計中,就是進程或線程之間的相互制約,也就是所謂的同步!
線程類比
一圖勝千言,我就不贅述了。
消費者線程想去找緩沖區要數據,先判斷緩沖區內有沒有數據,如果沒有,消費者就拿不到,這個線程就等待,直到:緩沖區內有數據。如果有,就從緩沖區將數據拿走。
生產者線程要去生產數據,先判斷緩沖區內有沒有空余位置,如果沒有,生產者就等待,直到:緩沖區內有空位,如果有,就生產數據,放入緩沖區。
阻塞隊列阻塞隊列正適合生產者消費者模型,當隊列滿時,入隊操作就會被阻塞,當隊列空時,出隊操作就會被阻塞。
入隊出隊的offer方法和poll方法與原隊列接口的方法相比,多了時間的參數。當發生阻塞時,如果超過了設置的時間,線程就會退出,畢竟如果最壞的情況,一直不滿足條件,也不能一直阻塞下去。
boolean offer(E e, long timeout, TimeUnit unit) throws InterruptedException; E poll(long timeout, TimeUnit unit) throws InterruptedException;實現類
ArrayBlockingQueue:數組實現的阻塞隊列。
LinkedBlockingQueue:鏈表實現的阻塞隊列。
PriorityBlockingQueue:優先級阻塞隊列。
雙向阻塞隊列這個簡單,就是同時實現了BlockingQueue和Deque接口。
java.util.concurrent包下只有一個雙向阻塞隊列的實現:LinkedBlockingDeque。
延時隊列延時隊列:DelayQueue,看這個類名,無疑了,此隊列定與時間有關。
當一個元素入隊時,它并不是馬上進入隊列,而是根據設定的時間延時之后再入隊。
假設offer一個元素,設置時間為10s,在10s內訪問隊列,是訪問不到元素的。
在延時之后,也就是10s之后,再去訪問,該元素才在隊列中。
使用場景
相關使用場景就是定時緩存。
HashMap和DelayQueue配合使用。用DelayQueue來存儲緩存的key,如果隊列中有元素,表示該key就已經過期。
然后再建一個線程去清理緩存,執行到poll方法時,使用不傳時間的方法,如果隊列為空,該線程就一直阻塞在這,不往下走。
隊列中有元素時,就說明有key過期了,線程繼續執行,然后元素出隊,根據相應的key移除緩存。
細節
延時隊列中存儲的元素需要實現Delayed接口。
public interface Delayed extends Comparable{ long getDelay(TimeUnit unit); }
getDelay方法返回剩余的延時時間,如果返回值大于0,表示還未到入隊時間。
同步隊列SynchronousQueue:同步隊列。
最好的解釋自然是官方文檔:A BlockingQueue in which each insert operation must wait for a corresponding remove operation by another thread, and vice versa.
這是一個阻塞隊列,它的特點是在執行插入操作時必須等待另一個線程的移除操作。什么意思呢?
通俗的來說就是買可樂不需要去超市了,我(消費者)直接和廠家(生產者)購買。
所以,生產者和消費者同時存在時,這個交易才能執行,兩方達成約定后,生產者生產可樂,賣給消費者。缺少任何一方另一方都會被阻塞,條件滿足時會喚醒對方繼續執行,這就是所謂的同步。
代碼層面講就是:put和take方法都被調用的時候,兩者才開始執行,并完成了數據的傳遞。
所以嚴格來說,雖然SynchronousQueue實現了隊列接口,但是它的目的卻并不是隊列,而是將生產者消費者線程配對。
轉移隊列LinkedTransferQueue:鏈式轉移隊列。雖然放在了最后,但是查閱相關文檔發現,實際的生產環境中,這個隊列最常用。
怎么轉移的呢?
消費者找隊列拿數據,如果沒有數據可用,就設置一個標志位,表示我這里期待著一個數據,然后消費者就開始等。
等著等著,直到生產者來了,判斷,如果有等著的,就直接把數據給它,實現了數據轉移。如果沒有呢?就去執行數據入隊相關的操作。
總結點開了阻塞隊列的源碼,發現線程安全是使用鎖實現的。
再看看面試問的東西:樂觀鎖、悲觀鎖、自旋鎖、偏向鎖、公平鎖,這都是寫啥東西呀?
吾生也有涯,而Java無涯。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/74322.html
摘要:知識點總結容器知識點總結容器接口與是在同一級別,都是繼承了接口。另一種隊列則是雙端隊列,支持在頭尾兩端插入和移除元素,主要包括。一個由鏈表結構組成的無界阻塞隊列。是一個阻塞的線程安全的隊列,底層實現也是使用鏈式結構。 Java知識點總結(Java容器-Queue) @(Java知識點總結)[Java, Java容器] Queue Queue接口與List、Set是在同一級別,都是繼承了...
摘要:語言在之前,提供的唯一的并發原語就是管程,而且之后提供的并發包,也是以管程技術為基礎的。但是管程更容易使用,所以選擇了管程。線程進入條件變量的等待隊列后,是允許其他線程進入管程的。并發編程里兩大核心問題互斥和同步,都可以由管程來幫你解決。 并發編程這個技術領域已經發展了半個世紀了。有沒有一種核心技術可以很方便地解決我們的并發問題呢?這個問題, 我會選擇 Monitor(管程)技術。Ja...
摘要:什么是阻塞隊列阻塞隊列是一個在隊列基礎上又支持了兩個附加操作的隊列。阻塞隊列的應用場景阻塞隊列常用于生產者和消費者的場景,生產者是向隊列里添加元素的線程,消費者是從隊列里取元素的線程。由鏈表結構組成的無界阻塞隊列。 什么是阻塞隊列? 阻塞隊列是一個在隊列基礎上又支持了兩個附加操作的隊列。 2個附加操作: 支持阻塞的插入方法:隊列滿時,隊列會阻塞插入元素的線程,直到隊列不滿。 支持阻塞的...
閱讀 3028·2021-11-24 10:21
閱讀 1598·2021-10-11 10:57
閱讀 2813·2021-09-22 15:24
閱讀 2674·2021-09-22 14:58
閱讀 2336·2019-08-30 13:16
閱讀 3487·2019-08-29 13:05
閱讀 3418·2019-08-29 12:14
閱讀 3456·2019-08-27 10:55