java并發(fā)編程工具類JUC之ArrayBlockingQueue
Java BlockingQueue接口java.util.concurrent.BlockingQueue表示一個(gè)可以存取元素,并且線程安全的隊(duì)列。換句話說,當(dāng)多線程同時(shí)從 JavaBlockingQueue中插入元素、獲取元素的時(shí)候,不會導(dǎo)致任何并發(fā)問題(元素被插入多次、處理多次等問題)。
從java BlockingQueue可以引申出一個(gè)概念:阻塞隊(duì)列,是指隊(duì)列本身可以阻塞線程向隊(duì)列里面插入元素,或者阻塞線程從隊(duì)列里面獲取元素。比如:當(dāng)一個(gè)線程嘗試去從一個(gè)空隊(duì)列里面獲取元素的時(shí)候,這個(gè)線程將被阻塞直到隊(duì)列內(nèi)元素?cái)?shù)量不再為空。當(dāng)然,線程是否會被阻塞取決于你調(diào)用什么方法從BlockingQueue獲取元素,有的方法會阻塞線程,有的方法會拋出異常等等,下文我們會詳細(xì)介紹。
類ArrayBlockingQueue是BlockingQueue接口的實(shí)現(xiàn)類,它是有界的阻塞隊(duì)列,內(nèi)部使用數(shù)組存儲隊(duì)列元素。這里的“有界”是指存儲容量存在上限,不能無限存儲元素。在同一時(shí)間內(nèi)存儲容量存在著一個(gè)上限值,這個(gè)上限制在初始實(shí)例化的時(shí)候指定,之后便不能修改了。
ArrayBlockingQueue內(nèi)部采用FIFO (First In, First Out)先進(jìn)先出的方法實(shí)現(xiàn)隊(duì)列數(shù)據(jù)的存取,隊(duì)首的元素是在隊(duì)列中保存時(shí)間最長的元素對象,隊(duì)尾的元素是在隊(duì)列中保存時(shí)間最短的元素對象。
下面的代碼說明如何初始化一個(gè)ArrayBlockingQueue,并向其中添加一個(gè)對象:
BlockingQueue queue = new ArrayBlockingQueue(1024);queue.put('1'); //向隊(duì)列中添加元素Object object = queue.take(); //從隊(duì)列中取出元素
BlockingQueue可以通過泛型來限定隊(duì)列中存儲數(shù)據(jù)的類型,下面的代碼以String為泛型,表示該隊(duì)列只能存儲String類型。
BlockingQueue<String> queue = new ArrayBlockingQueue<String>(1024);queue.put('1');String string = queue.take();實(shí)現(xiàn)一個(gè)生產(chǎn)消費(fèi)的實(shí)例
在前面的文章中我們曾經(jīng)講過:BlockingQueue經(jīng)常被用于生產(chǎn)消費(fèi)的緩沖隊(duì)列。下面我們就使用ArrayBlockingQueue來真正的實(shí)現(xiàn)一個(gè)生產(chǎn)消費(fèi)的例子。
類BlockingQueueExample開啟兩個(gè)獨(dú)立線程,一個(gè)是Producer生產(chǎn)者線程,負(fù)責(zé)向隊(duì)列中添加數(shù)據(jù);另一個(gè)是Consumer消費(fèi)者線程,負(fù)責(zé)從隊(duì)列中取出數(shù)據(jù)進(jìn)行處理。
public class BlockingQueueExample { public static void main(String[] args) throws Exception {//使用ArrayBlockingQueue初始化一個(gè)BlockingQueue,指定容量的上限為1024BlockingQueue queue = new ArrayBlockingQueue(1024);Producer producer = new Producer(queue); //生產(chǎn)者Consumer consumer = new Consumer(queue); //消費(fèi)者new Thread(producer).start(); //開啟生產(chǎn)者線程new Thread(consumer).start(); //開啟消費(fèi)者線程Thread.sleep(4000); }}
類Producer為生產(chǎn)者,每隔10秒鐘使用put()方法向隊(duì)列中放入一個(gè)對象,放入三次。在這10秒的間隔內(nèi),隊(duì)列數(shù)據(jù)被消費(fèi)者取走之后將導(dǎo)致消費(fèi)者線程阻塞。
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();} }}
下面的代碼是消費(fèi)者類Consumer,它從隊(duì)列中獲取待處理的元素對象,并調(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秒打印一次,因?yàn)槠渲衪ake()方法在隊(duì)列內(nèi)沒有元素可以取到的時(shí)候,會阻塞當(dāng)前的消費(fèi)者線程,讓其處于等待狀態(tài),這個(gè)方法我們在上一節(jié)介紹BlockingQueue的時(shí)候就已經(jīng)進(jìn)行過說明。
以上就是java并發(fā)編程工具類JUC之ArrayBlockingQueue的詳細(xì)內(nèi)容,更多關(guān)于java并發(fā)編程工具類ArrayBlockingQueue的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. Gitlab CI-CD自動化部署SpringBoot項(xiàng)目的方法步驟2. 淺談SpringMVC jsp前臺獲取參數(shù)的方式 EL表達(dá)式3. ajax請求添加自定義header參數(shù)代碼4. JS sort方法基于數(shù)組對象屬性值排序5. 使用Python和百度語音識別生成視頻字幕的實(shí)現(xiàn)6. ASP中解決“對象關(guān)閉時(shí),不允許操作。”的詭異問題……7. JAVA上加密算法的實(shí)現(xiàn)用例8. 基于javascript處理二進(jìn)制圖片流過程詳解9. ASP刪除img標(biāo)簽的style屬性只保留src的正則函數(shù)10. Django-migrate報(bào)錯問題解決方案
