JAVA CountDownLatch與thread-join()的區(qū)別解析
今天學(xué)習(xí)CountDownLatch這個(gè)類,作用感覺和join很像,然后就百度了一下,看了他們之間的區(qū)別。所以在此記錄一下。
首先來看一下join,在當(dāng)前線程中,如果調(diào)用某個(gè)thread的join方法,那么當(dāng)前線程就會(huì)被阻塞,直到thread線程執(zhí)行完畢,當(dāng)前線程才能繼續(xù)執(zhí)行。join的原理是,不斷的檢查thread是否存活,如果存活,那么讓當(dāng)前線程一直wait,直到thread線程終止,線程的this.notifyAll 就會(huì)被調(diào)用。
我們來看一下這個(gè)應(yīng)用場(chǎng)景:假設(shè)現(xiàn)在公司有三個(gè)員工A,B,C,他們要開會(huì)。但是A需要等B,C準(zhǔn)備好之后再才能開始,B,C需要同時(shí)準(zhǔn)備。我們先用join模擬上面的場(chǎng)景。
Employee.java:
public class Employee extends Thread{ private String employeeName; private long time; public Employee(String employeeName,long time){ this.employeeName = employeeName; this.time = time; } @Override public void run() { try { System.out.println(employeeName+ "開始準(zhǔn)備"); Thread.sleep(time); System.out.println(employeeName+" 準(zhǔn)備完成"); } catch (Exception e) { e.printStackTrace(); } } }
JoinTest.java:
public class JoinTest { public static void main(String[] args) throws InterruptedException { Employee a = new Employee("A", 3000); Employee b = new Employee("B", 3000); Employee c = new Employee("C", 4000); b.start(); c.start(); b.join(); c.join(); System.out.println("B,C準(zhǔn)備完成"); a.start(); } }
最后輸出結(jié)果如下:
C開始準(zhǔn)備 B開始準(zhǔn)備 B 準(zhǔn)備完成 C 準(zhǔn)備完成 B,C準(zhǔn)備完成 A開始準(zhǔn)備 A 準(zhǔn)備完成
可以看到,A總是在B,C準(zhǔn)備完成之后才開始執(zhí)行的。
CountDownLatch中我們主要用到兩個(gè)方法一個(gè)是await()方法,調(diào)用這個(gè)方法的線程會(huì)被阻塞,另外一個(gè)是countDown()方法,調(diào)用這個(gè)方法會(huì)使計(jì)數(shù)器減一,當(dāng)計(jì)數(shù)器的值為0時(shí),因調(diào)用await()方法被阻塞的線程會(huì)被喚醒,繼續(xù)執(zhí)行。
接下來,我們用CountDownLatch來模擬一下。
Employee.java:
public class Employee extends Thread{ private String employeeName; private long time; private CountDownLatch countDownLatch; public Employee(String employeeName,long time, CountDownLatch countDownLatch){ this.employeeName = employeeName; this.time = time; this.countDownLatch = countDownLatch; } @Override public void run() { try { System.out.println(employeeName+ "開始準(zhǔn)備"); Thread.sleep(time); System.out.println(employeeName+" 準(zhǔn)備完成"); countDownLatch.countDown(); } catch (Exception e) { e.printStackTrace(); } } }
CountDownLatchTest.java:
public class CountDownLatchTest { public static void main(String[] args) throws InterruptedException { CountDownLatch countDownLatch = new CountDownLatch(2); Employee a = new Employee("A", 3000,countDownLatch); Employee b = new Employee("B", 3000,countDownLatch); Employee c = new Employee("C", 4000,countDownLatch); b.start(); c.start(); countDownLatch.await(); System.out.println("B,C準(zhǔn)備完成"); a.start(); } }
輸出結(jié)果如下:
B開始準(zhǔn)備 C開始準(zhǔn)備 B 準(zhǔn)備完成 C 準(zhǔn)備完成 B,C準(zhǔn)備完成 A開始準(zhǔn)備 A 準(zhǔn)備完成
上面可以看到,CountDownLatch與join都能夠模擬上述的場(chǎng)景,那么他們有什么不同呢?這時(shí)候我們?cè)囅肓硗庖粋€(gè)場(chǎng)景就能看到他們的區(qū)別了。
假設(shè)A,B,C的工作都分為兩個(gè)階段,A只需要等待B,C各自完成他們工作的第一個(gè)階段就可以執(zhí)行了。
我們來修改一下Employee類:
public class Employee extends Thread{ private String employeeName; private long time; private CountDownLatch countDownLatch; public Employee(String employeeName,long time, CountDownLatch countDownLatch){ this.employeeName = employeeName; this.time = time; this.countDownLatch = countDownLatch; } @Override public void run() { try { System.out.println(employeeName+ " 第一階段開始準(zhǔn)備"); Thread.sleep(time); System.out.println(employeeName+" 第一階段準(zhǔn)備完成"); countDownLatch.countDown(); System.out.println(employeeName+ " 第二階段開始準(zhǔn)備"); Thread.sleep(time); System.out.println(employeeName+" 第二階段準(zhǔn)備完成"); } catch (Exception e) { e.printStackTrace(); } } }
CountDownLatchTest類不需要做修改,輸出結(jié)果入下:
B 第一階段開始準(zhǔn)備 C 第一階段開始準(zhǔn)備 B 第一階段準(zhǔn)備完成 B 第二階段開始準(zhǔn)備 C 第一階段準(zhǔn)備完成 C 第二階段開始準(zhǔn)備 B,C第一階段準(zhǔn)備完成 A 第一階段開始準(zhǔn)備 B 第二階段準(zhǔn)備完成 A 第一階段準(zhǔn)備完成 A 第二階段開始準(zhǔn)備 C 第二階段準(zhǔn)備完成 A 第二階段準(zhǔn)備完成
從結(jié)果可以看出,A在B,C第一階段準(zhǔn)備完成的時(shí)候就開始執(zhí)行了,不需要等到第二階段準(zhǔn)備完成。這種場(chǎng)景下,用join是沒法實(shí)現(xiàn)的。
總結(jié):調(diào)用join方法需要等待thread執(zhí)行完畢才能繼續(xù)向下執(zhí)行,而CountDownLatch只需要檢查計(jì)數(shù)器的值為零就可以繼續(xù)向下執(zhí)行,相比之下,CountDownLatch更加靈活一些,可以實(shí)現(xiàn)一些更加復(fù)雜的業(yè)務(wù)場(chǎng)景。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
java:無法訪問org.springframework.boot.SpringApplication的解決方法
這篇文章主要給大家介紹了關(guān)于java:無法訪問org.springframework.boot.SpringApplication的解決方法,文中通過實(shí)例代碼將解決的辦法介紹的非常詳細(xì),需要的朋友可以參考下2023-01-01SpringMVC框架搭建idea2021.3.2操作數(shù)據(jù)庫的示例詳解
這篇文章主要介紹了SpringMVC框架搭建idea2021.3.2操作數(shù)據(jù)庫,本文通過示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-04-04Mybatis之類型處理器TypeHandler的作用與自定義方式
這篇文章主要介紹了Mybatis之類型處理器TypeHandler的作用與自定義方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-04-04Java使用黑盒方式模擬實(shí)現(xiàn)內(nèi)網(wǎng)穿透
這篇文章主要介紹了Java使用黑盒方式模擬實(shí)現(xiàn)內(nèi)網(wǎng)穿透,內(nèi)網(wǎng)穿透,也即 NAT 穿透,進(jìn)行 NAT 穿透是為了使具有某一個(gè)特定源 IP 地址和源端口號(hào)的數(shù)據(jù)包不被 NAT 設(shè)備屏蔽而正確路由到內(nèi)網(wǎng)主機(jī),需要的朋友可以參考下2023-05-05Java POI讀取excel中數(shù)值精度損失問題解決
這篇文章主要介紹了Java POI讀取excel中數(shù)值精度損失問題解決,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04MyBatis游標(biāo)Cursor的正確使用和百萬數(shù)據(jù)傳輸?shù)膬?nèi)存測(cè)試
這篇文章主要介紹了MyBatis游標(biāo)Cursor的正確使用和百萬數(shù)據(jù)傳輸?shù)膬?nèi)存測(cè)試,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01java實(shí)現(xiàn)字符串和日期類型相互轉(zhuǎn)換的方法
這篇文章主要介紹了java實(shí)現(xiàn)字符串和日期類型相互轉(zhuǎn)換的方法,涉及java針對(duì)日期與字符串的轉(zhuǎn)換與運(yùn)算相關(guān)操作技巧,需要的朋友可以參考下2017-02-02