Java多線程CountDownLatch的實(shí)現(xiàn)
介紹
CountDownLatch是一個(gè)同步輔助類,它允許一個(gè)或多個(gè)線程一直等待直到其他線程執(zhí)行完畢才開始執(zhí)行。
用給定的計(jì)數(shù)初始化CountDownLatch,其含義是要被等待執(zhí)行完的線程個(gè)數(shù)。
每次調(diào)用CountDown(),計(jì)數(shù)減1
主程序執(zhí)行到await()函數(shù)會(huì)阻塞等待線程的執(zhí)行,直到計(jì)數(shù)為0
場(chǎng)景
1:多線程讀取批量文件, 并且讀取完成之后匯總處理
2:多線程讀取Excel多個(gè)sheet,讀取完成之后獲取匯總獲取的結(jié)果
3:多個(gè)人一起一起來吃飯,主人等待客人到來,客人一個(gè)個(gè)從不同地方來到飯店,主人需要等到所有人都到來之后,才能開飯
4:汽車站,所有乘客都從不同的地方趕到汽車站,必須等到所有乘客都到了,汽車才會(huì)出發(fā),(如果設(shè)置了超時(shí)等待,那么當(dāng)某個(gè)時(shí)間點(diǎn)到了,汽車也出發(fā))
5: 百米賽跑,4名運(yùn)動(dòng)員選手到達(dá)場(chǎng)地等待裁判口令,裁判一聲口令,選手聽到后同時(shí)起跑,當(dāng)所有選手到達(dá)終點(diǎn),裁判進(jìn)行匯總排名
6: 4名選手進(jìn)行大眾投票,投票現(xiàn)場(chǎng)有500個(gè)票, 投票時(shí)間是30分鐘,500個(gè)票投完結(jié)束投票,或者投票時(shí)間到了也結(jié)束投票
作用:可以用來確保某些活動(dòng)直到其他活動(dòng)都完成后才繼續(xù)執(zhí)行。
CountDownLatch非常適合于對(duì)任務(wù)進(jìn)行拆分,使其并行執(zhí)行,比如某個(gè)任務(wù)執(zhí)行2s,其對(duì)數(shù)據(jù)的請(qǐng)求可以分為五個(gè)部分,那么就可以將這個(gè)任務(wù)拆分為5個(gè)子任務(wù),分別交由五個(gè)線程執(zhí)行,執(zhí)行完成之后再由主線程進(jìn)行匯總,此時(shí),總的執(zhí)行時(shí)間將決定于執(zhí)行最慢的任務(wù),平均來看,還是大大減少了總的執(zhí)行時(shí)間。
注意事項(xiàng):
- 使用CountDownLatch必須確保計(jì)數(shù)器數(shù)量與子線程數(shù)量一致,且countDown必須要執(zhí)行,否則出現(xiàn)計(jì)數(shù)器不為0,導(dǎo)致主線程一致等待的情況
- 在執(zhí)行任務(wù)的線程中,使用了try...finally結(jié)構(gòu),該結(jié)構(gòu)可以保證創(chuàng)建的線程發(fā)生異常時(shí)CountDownLatch.countDown()方法也會(huì)執(zhí)行,也就保證了主線程不會(huì)一直處于等待狀態(tài)。
案例(模擬乘客登機(jī)的場(chǎng)景)
? ? public static void main(String[] args) {
? ? ? ? CountDownLatchUtils.initialize("da1", 5);
? ? ? ? List<String> list=new CopyOnWriteArrayList<>();
? ? ? ? ExecutorUtils.create(()->{
? ? ? ? ? ? System.out.println("張三正在馬鞍山,準(zhǔn)備趕到南京坐飛機(jī),需要1小時(shí)的車程到機(jī)場(chǎng)");
? ? ? ? ? ? SleepTools.second(4);
? ? ? ? ? ? list.add("張三");
? ? ? ? ? ? CountDownLatchUtils.countDown("da1");
? ? ? ? });
? ? ? ? ExecutorUtils.create(()->{
? ? ? ? ? ? System.out.println("李四正在徐州,準(zhǔn)備趕到南京坐飛機(jī),需要5小時(shí)的車程到機(jī)場(chǎng)");
? ? ? ? ? ? SleepTools.second(15);
? ? ? ? ? ? list.add("李四");
? ? ? ? ? ? CountDownLatchUtils.countDown("da1");
? ? ? ? });
? ? ? ? ExecutorUtils.create(()->{
? ? ? ? ? ? System.out.println("王五正在蕪湖,準(zhǔn)備趕到南京坐飛機(jī),需要2小時(shí)的車程到機(jī)場(chǎng)");
? ? ? ? ? ? SleepTools.second(9);
? ? ? ? ? ? list.add("王五");
? ? ? ? ? ? CountDownLatchUtils.countDown("da1");
? ? ? ? });
? ? ? ? ExecutorUtils.create(()->{
? ? ? ? ? ? //飛機(jī)起飛
? ? ? ? ? ? CountDownLatchUtils.await("da1",10); ?//這里先模擬10秒 360秒=1小時(shí)
? ? ? ? ? ? System.out.println("南京祿口機(jī)場(chǎng)_機(jī)長(zhǎng)啟動(dòng)飛機(jī)起飛");
? ? ? ? ? ? //當(dāng)前航班已到乘客
? ? ? ? ? ? System.out.println("當(dāng)前航班已到乘客"+list);
? ? ? ? });
? ? }以上代碼都是我封裝后的,主要是看邏輯就行了
張三正在馬鞍山,準(zhǔn)備趕到南京坐飛機(jī),需要1小時(shí)的車程到機(jī)場(chǎng)
李四正在徐州,準(zhǔn)備趕到南京坐飛機(jī),需要5小時(shí)的車程到機(jī)場(chǎng)
王五正在蕪湖,準(zhǔn)備趕到南京坐飛機(jī),需要2小時(shí)的車程到機(jī)場(chǎng)
南京祿口機(jī)場(chǎng)_機(jī)長(zhǎng)啟動(dòng)飛機(jī)起飛
當(dāng)前航班已到乘客[張三, 王五]
到此這篇關(guān)于Java多線程CountDownLatch的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)Java多線程CountDownLatch內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java啟用Azure Linux虛擬機(jī)診斷設(shè)置
這篇文章主要介紹了Java啟用Azure Linux虛擬機(jī)診斷設(shè)置,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05
lombok注解@Data使用在繼承類上時(shí)出現(xiàn)警告的問題及解決
Lombok的@Data注解簡(jiǎn)化了實(shí)體類代碼,但在子類中使用時(shí)會(huì)出現(xiàn)警告,指出equals和hashCode方法不會(huì)考慮父類屬性,解決方法有兩種:一是在父類上使用@EqualsAndHashCode(callSuper=true)注解;二是通過配置lombok.config文件,均能有效解決警告問題2024-10-10
Java 如何將網(wǎng)絡(luò)資源url轉(zhuǎn)化為File文件
這篇文章主要介紹了Java 如何將網(wǎng)絡(luò)資源url轉(zhuǎn)化為File文件的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
Java實(shí)現(xiàn)有限狀態(tài)機(jī)的推薦方案分享
有限狀態(tài)機(jī)又稱有限狀態(tài)自動(dòng)機(jī),簡(jiǎn)稱狀態(tài)機(jī),是表示有限個(gè)狀態(tài)以及在這些狀態(tài)之間的轉(zhuǎn)移和動(dòng)作等行為的數(shù)學(xué)模型,這篇文章主要給大家介紹了關(guān)于Java實(shí)現(xiàn)有限狀態(tài)機(jī)的推薦方案,需要的朋友可以參考下2021-11-11

