Java中Queue以及Deque用法示例詳解
一、核心概念與繼承體系

二、Queue 核心方法與實現(xiàn)
1. 核心操作:
| 方法 | 說明 | 異常處理 |
|---|---|---|
offer(e) | 添加元素(推薦) | 失敗返回false |
add(e) | 添加元素 | 失敗拋IllegalStateException |
poll() | 移除并返回隊頭元素 | 空隊列返回null |
remove() | 移除并返回隊頭元素 | 空隊列拋NoSuchElementException |
peek() | 查看隊頭元素(不刪除) | 空隊列返回null |
element() | 查看隊頭元素(不刪除) | 空隊列拋NoSuchElementException |
2. 常用實現(xiàn)類:
LinkedList:基于鏈表,支持null元素
PriorityQueue:基于堆的優(yōu)先級隊列(自然序/Comparator)
ArrayBlockingQueue:固定大小的阻塞隊列(線程安全)
LinkedBlockingQueue:可選有界阻塞隊列(線程安全)
三、Deque 雙端隊列擴展
1. 核心操作增強:
// 頭部操作 offerFirst(e) // 頭部插入 pollFirst() // 頭部移除 peekFirst() // 查看頭部 // 尾部操作 offerLast(e) // 尾部插入(等價于offer) pollLast() // 尾部移除 peekLast() // 查看尾部 // 棧操作 push(e) // = offerFirst(e) pop() // = removeFirst()
2. 作為隊列使用(FIFO)的API:
// 隊列操作(FIFO:先進先出) offerLast(e) 或 offer(e) // 入隊(尾部添加) pollFirst() 或 poll() // 出隊(頭部移除) peekFirst() 或 peek() // 查看隊頭
3. 作為棧使用(LIFO)的API:
// 棧操作(LIFO:后進先出) push(e) // 入棧 = addFirst(e) pop() // 出棧 = removeFirst() peekFirst() // 查看棧頂
4. API使用對照表:
| 操作 | 隊列模式(FIFO) | 棧模式(LIFO) | 等效方法 |
|---|---|---|---|
| 添加元素 | offerLast(e) / offer(e) | push(e) | addFirst(e)(棧) |
addLast(e)(隊列) | |||
| 移除元素 | pollFirst() / poll() | pop() | removeFirst() |
| 查看元素 | peekFirst() / peek() | peekFirst() | getFirst() |
5. 代碼示例:
Deque<String> deque = new ArrayDeque<>();
// 作為隊列使用(FIFO)
deque.offerLast("A"); // 隊尾添加
deque.offerLast("B");
System.out.println(deque.pollFirst()); // A(隊頭移除)
// 作為棧使用(LIFO)
deque.push("C"); // 入棧
deque.push("D");
System.out.println(deque.pop()); // D(出棧)
// 混合操作(不推薦但可能)
deque.offerLast("E"); // 隊尾添加(隊列操作)
deque.push("F"); // 棧頂添加(棧操作)
System.out.println(deque.pollFirst()); // F(混合操作結(jié)果)6. 常用實現(xiàn)類:
ArrayDeque:基于循環(huán)數(shù)組(默認容量16,性能最優(yōu))
LinkedList:基于雙向鏈表(支持索引訪問)
LinkedBlockingDeque:線程安全阻塞雙端隊列
四、與其他集合類對比
| 特性 | Queue/Deque | List | Set | Map |
|---|---|---|---|---|
| 數(shù)據(jù)結(jié)構(gòu) | 線性序列 | 線性序列 | 哈希表/樹 | 鍵值對 |
| 元素順序 | FIFO/LIFO/優(yōu)先級 | 插入順序/索引 | 無序/排序 | 無序/鍵排序 |
| 重復(fù)元素 | 允許 | 允許 | 不允許 | 值允許,鍵不允許 |
| 空值支持 | 部分實現(xiàn)支持 | 允許 | HashSet允許 | HashMap允許值 |
| 訪問方式 | 端點訪問 | 索引/迭代器 | 迭代器 | 鍵訪問 |
| 典型實現(xiàn) | ArrayDeque, PriorityQueue | ArrayList, LinkedList | HashSet, TreeSet | HashMap, TreeMap |
五、使用場景與最佳實踐
1. 隊列場景:
任務(wù)調(diào)度:
ThreadPoolExecutor使用BlockingQueue消息傳遞:生產(chǎn)者-消費者模式
廣度優(yōu)先搜索(BFS)
2. 雙端隊列場景:
撤銷操作棧:
ArrayDeque替代Stack滑動窗口算法
工作竊取算法(Work Stealing)
3. 選擇指南:

六、常見問題
Q1:Queue和Deque的主要區(qū)別是什么?
A:
功能定位:
Queue是標準FIFO隊列(尾部添加,頭部移除)
Deque是雙端隊列,擴展了Queue,支持兩端操作操作能力:
Queue只有隊頭出隊(poll)、隊尾入隊(offer)
Deque增加offerFirst/pollFirst等雙端操作方法棧功能:
Deque可直接作為棧使用(push/pop方法)
Queue沒有原生棧操作支持
Q2:ArrayDeque和LinkedList如何選擇?
A:
ArrayDeque:
基于循環(huán)數(shù)組,內(nèi)存連續(xù)
兩端操作時間復(fù)雜度O(1)
隨機訪問更快,CPU緩存友好
推薦場景:大多數(shù)隊列/棧需求(默認選擇)
LinkedList:
基于雙向鏈表,內(nèi)存分散
支持
List接口的索引訪問插入刪除中間元素更高效
推薦場景:
需要同時使用隊列和列表功能
需要頻繁在中間位置插入/刪除
Q3:阻塞隊列是什么?常用實現(xiàn)有哪些?
A:
阻塞隊列:當隊列滿時阻塞生產(chǎn)者,隊列空時阻塞消費者(
BlockingQueue接口)常用實現(xiàn):
ArrayBlockingQueue:數(shù)組實現(xiàn)的有界隊列
LinkedBlockingQueue:鏈表實現(xiàn)的可選有界隊列
PriorityBlockingQueue:帶優(yōu)先級的無界阻塞隊列
SynchronousQueue:不存儲元素的直接傳遞隊列
Q4:Deque如何替代Stack?
A:
Java官方推薦用
Deque替代Stack類轉(zhuǎn)換方式:
Deque<Integer> stack = new ArrayDeque<>(); stack.push(1); // 入棧 = addFirst() int top = stack.peek(); // 查看棧頂 = peekFirst() int pop = stack.pop(); // 出棧 = removeFirst()優(yōu)勢:
避免
Stack的同步開銷(Vector實現(xiàn))更統(tǒng)一的集合API
更好的性能(特別是
ArrayDeque)
Q5:PriorityQueue的排序原理?
A:
基于堆數(shù)據(jù)結(jié)構(gòu)(默認最小堆)
排序規(guī)則:
自然排序:元素實現(xiàn)
Comparable定制排序:構(gòu)造時傳入
Comparator特點:
隊頭總是當前最值元素
入隊/出隊時間復(fù)雜度O(log n)
不支持
null元素
Q6:如何使用Deque同時作為隊列和棧?
A:
Deque可以同時支持隊列和棧操作,但必須避免混用API:
隊列模式(FIFO):固定使用尾部添加+頭部移除組合
// 推薦API組合 deque.offer(e); // 入隊(尾部) String item = deque.poll(); // 出隊(頭部)
棧模式(LIFO):固定使用頭部添加+頭部移除組合
// 推薦API組合 deque.push(e); // 入棧(頭部) String top = deque.pop(); // 出棧(頭部)
危險操作:混用API會導(dǎo)致數(shù)據(jù)順序混亂
// 錯誤示例(導(dǎo)致數(shù)據(jù)順序不可預(yù)測) deque.push("A"); // 棧操作(頭部插入) deque.offer("B"); // 隊列操作(尾部插入) // 此時隊列:A<-B,但棧頂是A
Q7:為什么Java推薦用Deque代替Stack類?
A:
除了之前提到的性能優(yōu)勢,API設(shè)計也更合理:
Stack的缺陷API:
// 老式Stack API(繼承自Vector) stack.addElement(e); // 非標準方法名 stack.insertElementAt(e, 0); // 危險的低效操作Deque的標準棧API:
deque.push(e); // 標準棧操作 deque.pop(); // 直觀的LIFO語義 deque.peek(); // 查看棧頂額外優(yōu)勢:Deque的棧操作時間復(fù)雜度均為O(1),而Stack的
insertElementAt(0)是O(n)
七、高頻面試進階問題
poll()和remove()有什么區(qū)別?
行為相同:移除并返回隊頭元素
空隊列時:
poll()返回null,remove()拋異常
ArrayDeque初始容量是多少?如何擴容?
默認初始容量16
擴容規(guī)則:加倍容量(16→32→64...)
重要特性:容量總是2的冪(位運算優(yōu)化)
阻塞隊列的put()和offer()區(qū)別?
// 阻塞方法(無限等待) void put(E e) throws InterruptedException; // 非阻塞方法 boolean offer(E e, long timeout, TimeUnit unit); // 限時等待 boolean offer(E e); // 立即返回
為什么LinkedList實現(xiàn)了List和Deque?
設(shè)計上支持多種訪問方式:
列表功能:索引訪問/中間插入
隊列功能:FIFO操作
雙端功能:兩端高效操作
體現(xiàn)了接口隔離原則
總結(jié)
到此這篇關(guān)于Java中Queue以及Deque用法的文章就介紹到這了,更多相關(guān)Java中Queue以及Deque內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java對象轉(zhuǎn)JSON時動態(tài)的增刪改查屬性詳解
這篇文章主要介紹了Java對象轉(zhuǎn)JSON時如何動態(tài)的增刪改查屬性的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-11-11
java并發(fā)編程中實現(xiàn)可見性的四種可行方案解析
這篇文章主要介紹了java并發(fā)編程中實現(xiàn)可見性的四種可行方案解析,使用關(guān)鍵字volatile和使用鎖(如synchronized關(guān)鍵字或者java.util.concurrent包中的鎖)來確保對共享變量的修改在多線程環(huán)境中能夠正確地被其他線程所觀察到,需要的朋友可以參考下2023-08-08
java-SSH2實現(xiàn)數(shù)據(jù)庫和界面的分頁
本文主要是介紹SSH2實現(xiàn)數(shù)據(jù)庫和界面的分頁的代碼,分頁在web應(yīng)用中是經(jīng)常要做的事情,實用性比較大,有需要的朋友可以來了解一下。2016-10-10
java使用Logback配置輸出日志內(nèi)容到文件示例代碼
這篇文章主要介紹了java?Logback輸出日志內(nèi)容到文件,要將logger.info的信息輸出到文件,您可以使用Logback配置,本文通過實例代碼給大家介紹的非常詳細,需要的朋友可以參考下2023-09-09
Hadoop?MapReduce實現(xiàn)單詞計數(shù)(Word?Count)
這篇文章主要為大家詳細介紹了如何利用Hadoop實現(xiàn)單詞計數(shù)(Word?Count)的MapReduce,文中的示例代碼講解詳細,感興趣的可以跟隨小編一起學習一下2023-05-05
Java子線程無法獲取Attributes的解決方法(最新推薦)
在Java多線程編程中,子線程無法直接獲取主線程設(shè)置的Attributes是一個常見問題,本文探討了這一問題的原因,并提供了兩種解決方案,對Java子線程無法獲取Attributes的解決方案感興趣的朋友一起看看吧2025-01-01
Java并發(fā)工具類CountDownLatch CyclicBarrier使用詳解
這篇文章主要為大家介紹了Java并發(fā)工具類CountDownLatch CyclicBarrier使用詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-06-06

