Java執行緒(篇外篇):阻塞佇列BlockingQueue
好久沒有寫文章了,這段時間事情比較雜,工作也比較雜亂,上週日剛搬完家,從自建房搬到了樓房,提升了一層生活品質,哈哈!不過昨天晚上在公交車上錢包被偷了,前段時間還丟個自行車,不得不感嘆,京城扒手真多,還無人處理。言歸正傳,這一段時間我的工作主要是改進公司的排程器,排程器排程執行緒池執行任務,生產者生產任務,消費者消費任務,那麼這時就需要一個任務佇列,生產者向佇列裡插入任務,消費者從佇列裡提取任務執行,排程器裡是通過BlockingQueue實現的佇列,隨後小查一下,下面看看BlockingQueue的原理及其方法。
BlockingQueue最終會有四種狀況,丟擲異常、返回特殊值、阻塞、超時,下表總結了這些方法:
丟擲異常 特殊值 阻塞 超時 插入 add(e)
offer(e)
put(e)
offer(e, time, unit)
移除 remove()
poll()
take()
poll(time, unit)
檢查 element()
peek()
不可用 不可用
BlockingQueue是個介面,有如下實現類:
1. ArrayBlockQueue:一個由陣列支援的有界阻塞佇列。此佇列按 FIFO(先進先出)原則對元素進行排序。建立其物件必須明確大小,像陣列一樣。
2. LinkedBlockQueue:一個可改變大小的阻塞佇列。此佇列按 FIFO(先進先出)原則對元素進行排序。建立其物件如果沒有明確大小,預設值是Integer.MAX_VALUE。連結佇列的吞吐量通常要高於基於陣列的佇列,但是在大多數併發應用程式中,其可預知的效能要低。
3. PriorityBlockingQueue:類似於LinkedBlockingQueue,但其所含物件的排序不是FIFO,而是依據物件的自然排序順序或者是建構函式所帶的Comparator決定的順序。
4. SynchronousQueue:同步佇列。同步佇列沒有任何容量,每個插入必須等待另一個執行緒移除,反之亦然。
下面使用ArrayBlockQueue來實現之前實現過的生產者消/費者模式,程式碼如下:
/** 定義一個盤子類,可以放雞蛋和取雞蛋 */
public class BigPlate {
/** 裝雞蛋的盤子,大小為5 */
private BlockingQueue<Object> eggs = new ArrayBlockingQueue<Object>(5);
/** 放雞蛋 */
public void putEgg(Object egg) {
try {
eggs.put(egg);// 向盤子末尾放一個雞蛋,如果盤子滿了,當前執行緒阻塞
} catch (InterruptedException e) {
e.printStackTrace();
}
// 下面輸出有時不準確,因為與put操作不是一個原子操作
System.out.println("放入雞蛋");
}
/** 取雞蛋 */
public Object getEgg() {
Object egg = null;
try {
egg = eggs.take();// 從盤子開始取一個雞蛋,如果盤子空了,當前執行緒阻塞
} catch (InterruptedException e) {
e.printStackTrace();
}
// 下面輸出有時不準確,因為與take操作不是一個原子操作
System.out.println("拿到雞蛋");
return egg;
}
/** 放雞蛋執行緒 */
static class AddThread extends Thread {
private BigPlate plate;
private Object egg = new Object();
public AddThread(BigPlate plate) {
this.plate = plate;
}
public void run() {
plate.putEgg(egg);
}
}
/** 取雞蛋執行緒 */
static class GetThread extends Thread {
private BigPlate plate;
public GetThread(BigPlate plate) {
this.plate = plate;
}
public void run() {
plate.getEgg();
}
}
public static void main(String[] args) {
BigPlate plate = new BigPlate();
// 先啟動10個放雞蛋執行緒
for(int i = 0; i < 10; i++) {
new Thread(new AddThread(plate)).start();
}
// 再啟動10個取雞蛋執行緒
for(int i = 0; i < 10; i++) {
new Thread(new GetThread(plate)).start();
}
}
}
執行結果:
放入雞蛋
放入雞蛋
放入雞蛋
放入雞蛋
放入雞蛋
拿到雞蛋
放入雞蛋
拿到雞蛋
拿到雞蛋
拿到雞蛋
放入雞蛋
放入雞蛋
放入雞蛋
拿到雞蛋
放入雞蛋
拿到雞蛋
拿到雞蛋
拿到雞蛋
拿到雞蛋
拿到雞蛋
從結果看,啟動10個放雞蛋執行緒和10個取雞蛋執行緒,前5個放入雞蛋的執行緒成功執行,到第6個,發現盤子滿了,阻塞住,這時切換到取雞蛋執行緒執行,成功實現了生產者/消費者模式。java.util.concurrent包是個強大的包!
本文來自:高爽|Coder,原文地址:http://blog.csdn.net/ghsau/article/details/8108292。
相關文章
- Java BlockingQueue 阻塞佇列[用於多執行緒]JavaBloC佇列執行緒
- java多執行緒:執行緒池原理、阻塞佇列Java執行緒佇列
- 阻塞佇列 BlockingQueue佇列BloC
- Java併發系列 — 阻塞佇列(BlockingQueue)Java佇列BloC
- 阻塞佇列BlockingQueue(三)--DelayQueue佇列BloC
- Java阻塞佇列執行緒集控制的實現Java佇列執行緒
- 深入淺出Java多執行緒(十三):阻塞佇列Java執行緒佇列
- java多執行緒8:阻塞佇列與Fork/Join框架Java執行緒佇列框架
- 執行緒池的阻塞佇列的理解執行緒佇列
- Java執行緒(篇外篇):執行緒本地變數ThreadLocalJava執行緒變數thread
- Java併發指南11:解讀 Java 阻塞佇列 BlockingQueueJava佇列BloC
- Java 阻塞佇列(BlockingQueue)的內部實現原理Java佇列BloC
- 探討阻塞佇列和執行緒池原始碼佇列執行緒原始碼
- 最全java多執行緒總結3——瞭解阻塞佇列和執行緒安全集合不Java執行緒佇列
- Java併發程式設計:4種執行緒池和緩衝佇列BlockingQueueJava程式設計執行緒佇列BloC
- java併發程式設計工具類JUC第一篇:BlockingQueue阻塞佇列Java程式設計BloC佇列
- 原始碼剖析ThreadPoolExecutor執行緒池及阻塞佇列原始碼thread執行緒佇列
- 阻塞佇列一——java中的阻塞佇列佇列Java
- java多執行緒:java佇列詳解Java執行緒佇列
- java執行緒池-工作佇列workQueueJava執行緒佇列
- 阻塞佇列 BlockingQueue(二)--ArrayBlockingQueue與LinkedBlockingQueue佇列BloC
- 多執行緒系列(十三) -一文帶你搞懂阻塞佇列執行緒佇列
- java 執行緒池執行緒忙碌且阻塞佇列也滿了時給一個拒接的詳細報告Java執行緒佇列
- 主佇列&主執行緒佇列執行緒
- java-執行緒池佇列飽和策略Java執行緒佇列
- 解讀 Java 併發佇列 BlockingQueueJava佇列BloC
- 多執行緒程式設計-分析阻塞佇列的原始碼實現執行緒程式設計佇列原始碼
- Java中的阻塞佇列Java佇列
- 佇列、阻塞佇列佇列
- Java幾種執行緒池及任務佇列Java執行緒佇列
- 多執行緒,執行緒類三種方式,執行緒排程,執行緒同步,死鎖,執行緒間的通訊,阻塞佇列,wait和sleep區別?執行緒佇列AI
- Java併發包原始碼學習系列:阻塞佇列BlockingQueue及實現原理分析Java原始碼佇列BloC
- iOS 多執行緒--GCD 序列佇列、併發佇列以及同步執行、非同步執行iOS執行緒GC佇列非同步
- 請教:java.exe執行緒阻塞Java執行緒
- 阻塞佇列佇列
- 執行緒的阻塞執行緒
- 多執行緒學習-Disruptor佇列執行緒佇列
- iOS-佇列.執行緒.任務iOS佇列執行緒