java并發(fā)編程工具類JUC之ArrayBlockingQueue
Java BlockingQueue接口java.util.concurrent.BlockingQueue表示一個可以存取元素,并且線程安全的隊列。換句話說,當多線程同時從 JavaBlockingQueue中插入元素、獲取元素的時候,不會導致任何并發(fā)問題(元素被插入多次、處理多次等問題)。
從java BlockingQueue可以引申出一個概念:阻塞隊列,是指隊列本身可以阻塞線程向隊列里面插入元素,或者阻塞線程從隊列里面獲取元素。比如:當一個線程嘗試去從一個空隊列里面獲取元素的時候,這個線程將被阻塞直到隊列內(nèi)元素數(shù)量不再為空。當然,線程是否會被阻塞取決于你調(diào)用什么方法從BlockingQueue獲取元素,有的方法會阻塞線程,有的方法會拋出異常等等,下文我們會詳細介紹。
類ArrayBlockingQueue是BlockingQueue接口的實現(xiàn)類,它是有界的阻塞隊列,內(nèi)部使用數(shù)組存儲隊列元素。這里的“有界”是指存儲容量存在上限,不能無限存儲元素。在同一時間內(nèi)存儲容量存在著一個上限值,這個上限制在初始實例化的時候指定,之后便不能修改了。
ArrayBlockingQueue內(nèi)部采用FIFO (First In, First Out)先進先出的方法實現(xiàn)隊列數(shù)據(jù)的存取,隊首的元素是在隊列中保存時間最長的元素對象,隊尾的元素是在隊列中保存時間最短的元素對象。
下面的代碼說明如何初始化一個ArrayBlockingQueue,并向其中添加一個對象:
BlockingQueue queue = new ArrayBlockingQueue(1024);
queue.put("1"); //向隊列中添加元素
Object object = queue.take(); //從隊列中取出元素
BlockingQueue可以通過泛型來限定隊列中存儲數(shù)據(jù)的類型,下面的代碼以String為泛型,表示該隊列只能存儲String類型。
BlockingQueue<String> queue = new ArrayBlockingQueue<String>(1024);
queue.put("1");
String string = queue.take();
實現(xiàn)一個生產(chǎn)消費的實例
在前面的文章中我們曾經(jīng)講過:BlockingQueue經(jīng)常被用于生產(chǎn)消費的緩沖隊列。下面我們就使用ArrayBlockingQueue來真正的實現(xiàn)一個生產(chǎn)消費的例子。
類BlockingQueueExample開啟兩個獨立線程,一個是Producer生產(chǎn)者線程,負責向隊列中添加數(shù)據(jù);另一個是Consumer消費者線程,負責從隊列中取出數(shù)據(jù)進行處理。
public class BlockingQueueExample {
public static void main(String[] args) throws Exception {
//使用ArrayBlockingQueue初始化一個BlockingQueue,指定容量的上限為1024
BlockingQueue queue = new ArrayBlockingQueue(1024);
Producer producer = new Producer(queue); //生產(chǎn)者
Consumer consumer = new Consumer(queue); //消費者
new Thread(producer).start(); //開啟生產(chǎn)者線程
new Thread(consumer).start(); //開啟消費者線程
Thread.sleep(4000);
}
}
類Producer為生產(chǎn)者,每隔10秒鐘使用put()方法向隊列中放入一個對象,放入三次。在這10秒的間隔內(nèi),隊列數(shù)據(jù)被消費者取走之后將導致消費者線程阻塞。
public class Producer implements Runnable{
protected BlockingQueue queue = null;
public Producer(BlockingQueue queue) {
this.queue = queue;
}
public void run() {
try {
queue.put("1");
Thread.sleep(10000);
queue.put("2");
Thread.sleep(10000);
queue.put("3");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
下面的代碼是消費者類Consumer,它從隊列中獲取待處理的元素對象,并調(diào)用System.out將其打印出來。
public class Consumer implements Runnable{
protected BlockingQueue queue = null;
public Consumer(BlockingQueue queue) {
this.queue = queue;
}
public void run() {
try {
System.out.println(queue.take());
System.out.println(queue.take());
System.out.println(queue.take());
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
上面的代碼打印結(jié)果是每隔10秒打印一次,因為其中take()方法在隊列內(nèi)沒有元素可以取到的時候,會阻塞當前的消費者線程,讓其處于等待狀態(tài),這個方法我們在上一節(jié)介紹BlockingQueue的時候就已經(jīng)進行過說明。
以上就是java并發(fā)編程工具類JUC之ArrayBlockingQueue的詳細內(nèi)容,更多關(guān)于java并發(fā)編程工具類ArrayBlockingQueue的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
springboot+dubbo+validation 進行rpc參數(shù)校驗的實現(xiàn)方法
這篇文章主要介紹了springboot+dubbo+validation 進行rpc參數(shù)校驗的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-09-09
Java調(diào)用ChatGPT的實現(xiàn)代碼
這篇文章主要介紹了Java調(diào)用ChatGPT的實現(xiàn)代碼,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-02-02
95%的Java程序員人都用不好Synchronized詳解
這篇文章主要為大家介紹了95%的Java程序員人都用不好Synchronized詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-03-03
基于Java HttpClient和Htmlparser實現(xiàn)網(wǎng)絡(luò)爬蟲代碼
這篇文章主要介紹了基于Java HttpClient和Htmlparser實現(xiàn)網(wǎng)絡(luò)爬蟲代碼的相關(guān)資料,需要的朋友可以參考下2015-12-12
解析Mybatis的insert方法返回數(shù)字-2147482646的解決
這篇文章主要介紹了解析Mybatis的insert方法返回數(shù)字-2147482646的解決,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-04-04
SpringBoot集成POI實現(xiàn)Excel導入導出的示例詳解
Apache?POI?是用Java編寫的免費開源的跨平臺的?Java?API,Apache?POI提供API給Java程序?qū)icrosoft?Office格式檔案讀和寫的功能。本文主要介紹通過SpringBoot集成POI工具實現(xiàn)Excel的導入和導出功能,需要的可以參考一下2022-07-07

