十年網(wǎng)站開(kāi)發(fā)經(jīng)驗(yàn) + 多家企業(yè)客戶(hù) + 靠譜的建站團(tuán)隊(duì)
量身定制 + 運(yùn)營(yíng)維護(hù)+專(zhuān)業(yè)推廣+無(wú)憂售后,網(wǎng)站問(wèn)題一站解決
本文實(shí)例講述了Java使用阻塞隊(duì)列控制線程通信的方法。分享給大家供大家參考,具體如下:
壽寧網(wǎng)站建設(shè)公司成都創(chuàng)新互聯(lián)公司,壽寧網(wǎng)站設(shè)計(jì)制作,有大型網(wǎng)站制作公司豐富經(jīng)驗(yàn)。已為壽寧1000+提供企業(yè)網(wǎng)站建設(shè)服務(wù)。企業(yè)網(wǎng)站搭建\成都外貿(mào)網(wǎng)站制作要多少錢(qián),請(qǐng)找那個(gè)售后服務(wù)好的壽寧做網(wǎng)站的公司定做!
一 點(diǎn)睛
阻塞隊(duì)列主要用在生產(chǎn)者/消費(fèi)者的場(chǎng)景,下面這幅圖展示了一個(gè)線程生產(chǎn)、一個(gè)線程消費(fèi)的場(chǎng)景:
負(fù)責(zé)生產(chǎn)的線程不斷的制造新對(duì)象并插入到阻塞隊(duì)列中,直到達(dá)到這個(gè)隊(duì)列的上限值。隊(duì)列達(dá)到上限值之后生產(chǎn)線程將會(huì)被阻塞,直到消費(fèi)的線程對(duì)這個(gè)隊(duì)列進(jìn)行消費(fèi)。同理,負(fù)責(zé)消費(fèi)的線程不斷的從隊(duì)列中消費(fèi)對(duì)象,直到這個(gè)隊(duì)列為空,當(dāng)隊(duì)列為空時(shí),消費(fèi)線程將會(huì)被阻塞,除非隊(duì)列中有新的對(duì)象被插入。
BlockingQueue的核心方法:
方法\行為 |
拋異常 |
特定的值 |
阻塞 |
超時(shí) |
插入方法 |
add(o) |
offer(o) |
put(o) |
offer(o, timeout, timeunit) |
移除方法 |
|
poll(),remove(o) |
take() |
poll(timeout, timeunit) |
獲取、不刪除元素 |
element() |
peek() |
|
|
行為解釋?zhuān)?/p>
1.拋異常:如果操作不能馬上進(jìn)行,則拋出異常。
2. 特定的值:如果操作不能馬上進(jìn)行,將會(huì)返回一個(gè)特殊的值,一般是true或者false。
3. 阻塞:如果操作不能馬上進(jìn)行,操作會(huì)被阻塞。
4. 超時(shí):如果操作不能馬上進(jìn)行,操作會(huì)被阻塞指定的時(shí)間,如果指定時(shí)間沒(méi)執(zhí)行,則返回一個(gè)特殊值,一般是true或者false。
插入方法:
刪除方法:
獲取、不刪除元素:
二 實(shí)戰(zhàn)1
1 代碼
import java.util.concurrent.*; public class BlockingQueueTest { public static void main(String[] args) throws Exception { // 定義一個(gè)長(zhǎng)度為2的阻塞隊(duì)列 BlockingQueuebq = new ArrayBlockingQueue<>(2); bq.put("Java"); // 與bq.add("Java"、bq.offer("Java")相同 bq.put("Java"); // 與bq.add("Java"、bq.offer("Java")相同 System.out.println("打印1"); bq.put("Java"); // ① 阻塞線程。 System.out.println("打印2"); } }
2 運(yùn)行
打印1
三 實(shí)戰(zhàn)2
1 代碼
import java.util.concurrent.*; public class BlockingQueueTest { public static void main(String[] args) throws Exception { // 定義一個(gè)長(zhǎng)度為2的阻塞隊(duì)列 BlockingQueuebq = new ArrayBlockingQueue<>(2); bq.put("Java"); // 與bq.add("Java"、bq.offer("Java")相同 bq.put("Java"); // 與bq.add("Java"、bq.offer("Java")相同 System.out.println("打印1"); //bq.put("Java"); // ① 阻塞線程。 System.out.println("打印2"); } }
2 運(yùn)行
打印1
打印2
四 實(shí)戰(zhàn)3
1 代碼
import java.util.concurrent.*; class Producer extends Thread { private BlockingQueuebq; public Producer(BlockingQueue bq) { this.bq = bq; } public void run() { String[] strArr = new String[] { "Java", "Struts", "Spring" }; for (int i = 0 ; i < 5 ; i++ ) { System.out.println(getName() + "生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!"); try { Thread.sleep(200); // 嘗試放入元素,如果隊(duì)列已滿(mǎn),線程被阻塞 bq.put(strArr[i % 3]); } catch (Exception ex){ex.printStackTrace();} System.out.println(getName() + "生產(chǎn)完成:" + bq); } } } class Consumer extends Thread { private BlockingQueue bq; public Consumer(BlockingQueue bq) { this.bq = bq; } public void run() { while(true) { System.out.println(getName() + "消費(fèi)者準(zhǔn)備消費(fèi)集合元素!"); try { Thread.sleep(200); // 嘗試取出元素,如果隊(duì)列已空,線程被阻塞 bq.take(); } catch (Exception ex){ex.printStackTrace();} System.out.println(getName() + "消費(fèi)完成:" + bq); } } } public class BlockingQueueTest2 { public static void main(String[] args) { // 創(chuàng)建一個(gè)容量為1的BlockingQueue BlockingQueue bq = new ArrayBlockingQueue<>(1); // 啟動(dòng)3條生產(chǎn)者線程 new Producer(bq).start(); new Producer(bq).start(); new Producer(bq).start(); // 啟動(dòng)一條消費(fèi)者線程 new Consumer(bq).start(); } }
2 運(yùn)行
Thread-1生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-2生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-0生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-0生產(chǎn)完成:[Java]
Thread-0生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)完成:[]
Thread-2生產(chǎn)完成:[Java]
Thread-2生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-3消費(fèi)完成:[Struts]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-2生產(chǎn)完成:[Struts]
Thread-2生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)完成:[]
Thread-0生產(chǎn)完成:[Struts]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-0生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)完成:[Java]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-1生產(chǎn)完成:[Java]
Thread-1生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)完成:[]
Thread-2生產(chǎn)完成:[Spring]
Thread-2生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-3消費(fèi)完成:[Java]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-2生產(chǎn)完成:[Java]
Thread-2生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)完成:[]
Thread-1生產(chǎn)完成:[Struts]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-1生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)完成:[Spring]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-1生產(chǎn)完成:[Spring]
Thread-1生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)完成:[]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-2生產(chǎn)完成:[Struts]
Thread-3消費(fèi)完成:[]
Thread-0生產(chǎn)完成:[Spring]
Thread-0生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-1生產(chǎn)完成:[Java]
Thread-1生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)完成:[Java]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-3消費(fèi)完成:[]
Thread-0生產(chǎn)完成:[Java]
Thread-0生產(chǎn)者準(zhǔn)備生產(chǎn)集合元素!
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-3消費(fèi)完成:[]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-0生產(chǎn)完成:[Struts]
Thread-3消費(fèi)完成:[]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
Thread-1生產(chǎn)完成:[Struts]
Thread-3消費(fèi)完成:[]
Thread-3消費(fèi)者準(zhǔn)備消費(fèi)集合元素!
更多java相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《Java進(jìn)程與線程操作技巧總結(jié)》、《Java數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Java操作DOM節(jié)點(diǎn)技巧總結(jié)》、《Java文件與目錄操作技巧匯總》和《Java緩存操作技巧匯總》
希望本文所述對(duì)大家java程序設(shè)計(jì)有所幫助。