日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > java >内容正文

java

Java主线程等待子线程、线程池

發布時間:2023/12/20 java 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Java主线程等待子线程、线程池 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
public class TestThread extends Thread { public void run() { System.out.println(this.getName() + "子線程開始"); try { // 子線程休眠五秒 Thread.sleep(5000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(this.getName() + "子線程結束"); } }

首先是一個線程,它執行完成需要5秒。

1、主線程等待一個子線程

public class Main { public static void main(String[] args) { long start = System.currentTimeMillis(); Thread thread = new TestThread(); thread.start(); long end = System.currentTimeMillis(); System.out.println("子線程執行時長:" + (end - start)); } }

在主線程中,需要等待子線程執行完成。但是執行上面的main發現并不是想要的結果:

子線程執行時長:0 Thread-0子線程開始 Thread-0子線程結束

很明顯主線程和子線程是并發執行的,主線程并沒有等待。

對于只有一個子線程,如果主線程需要等待子線程執行完成,再繼續向下執行,可以使用Thread的join()方法。join()方法會阻塞主線程繼續向下執行。

public class Main { public static void main(String[] args) { long start = System.currentTimeMillis(); Thread thread = new TestThread(); thread.start(); try { thread.join(); } catch (InterruptedException e) { e.printStackTrace(); } long end = System.currentTimeMillis(); System.out.println("子線程執行時長:" + (end - start)); } }

執行結果:

Thread-0子線程開始 Thread-0子線程結束 子線程執行時長:5000

注意:join()要在start()方法之后調用。

2、主線程等待多個子線程

比如主線程需要等待5個子線程。這5個線程之間是并發執行。

public class Main { public static void main(String[] args) { long start = System.currentTimeMillis(); for(int i = 0; i < 5; i++) { Thread thread = new TestThread(); thread.start(); try { thread.join(); } catch (InterruptedException e) { e.printStackTrace(); } } long end = System.currentTimeMillis(); System.out.println("子線程執行時長:" + (end - start)); } }

在上面的代碼套上一個for循環,執行結果:

Thread-0子線程開始 Thread-0子線程結束 Thread-1子線程開始 Thread-1子線程結束 Thread-2子線程開始 Thread-2子線程結束 Thread-3子線程開始 Thread-3子線程結束 Thread-4子線程開始 Thread-4子線程結束 子線程執行時長:25000

由于thread.join()阻塞了主線程繼續執行,導致for循環一次就需要等待一個子線程執行完成,而下一個子線程不能立即start(),5個子線程不能并發。

要想子線程之間能并發執行,那么需要在所有子線程start()后,在執行所有子線程的join()方法。

public class Main { public static void main(String[] args) { long start = System.currentTimeMillis(); List<Thread> list = new ArrayList<Thread>(); for(int i = 0; i < 5; i++) { Thread thread = new TestThread(); thread.start(); list.add(thread); } try { for(Thread thread : list) { thread.join(); } } catch (InterruptedException e) { e.printStackTrace(); } long end = System.currentTimeMillis(); System.out.println("子線程執行時長:" + (end - start)); } }

執行結果:

Thread-0子線程開始 Thread-3子線程開始 Thread-1子線程開始 Thread-2子線程開始 Thread-4子線程開始 Thread-3子線程結束 Thread-0子線程結束 Thread-2子線程結束 Thread-1子線程結束 Thread-4子線程結束 子線程執行時長:5000

3、主線程等待多個子線程(CountDownLatch實現)

CountDownLatch是Java.util.concurrent中的一個同步輔助類,可以把它看做一個倒數計數器,就像神舟十號發射時倒數:10,9,8,7….2,1,0,走你。初始化時先設置一個倒數計數初始值,每調用一次countDown()方法,倒數值減一,await()方法會阻塞當前進程,直到倒數至0。

同樣還是主線程等待5個并發的子線程。修改上面的代碼,在主線程中,創建一個初始值為5的CountDownLatch,并傳給每個子線程,在每個子線程最后調用countDown()方法對倒數器減1,當5個子線程等執行完成,那么CountDownLatch也就倒數完成,主線程調用await()方法等待5個子線程執行完成。

修改MyThread接收傳入的CountDownLatch:

public class TestThread extends Thread { private CountDownLatch countDownLatch; public TestThread(CountDownLatch countDownLatch) { this.countDownLatch = countDownLatch; } public void run() { System.out.println(this.getName() + "子線程開始"); try { // 子線程休眠五秒 Thread.sleep(5000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(this.getName() + "子線程結束"); // 倒數器減1 countDownLatch.countDown(); } }

修改main:

public class Main { public static void main(String[] args) { long start = System.currentTimeMillis(); // 創建一個初始值為5的倒數計數器 CountDownLatch countDownLatch = new CountDownLatch(5); for(int i = 0; i < 5; i++) { Thread thread = new TestThread(countDownLatch); thread.start(); } try { // 阻塞當前線程,直到倒數計數器倒數到0 countDownLatch.await(); } catch (InterruptedException e) { e.printStackTrace(); } long end = System.currentTimeMillis(); System.out.println("子線程執行時長:" + (end - start)); } }

執行結果:

Thread-0子線程開始 Thread-2子線程開始 Thread-1子線程開始 Thread-3子線程開始 Thread-4子線程開始 Thread-2子線程結束 Thread-4子線程結束 Thread-1子線程結束 Thread-0子線程結束 Thread-3子線程結束 子線程執行時長:5000

注意:如果子線程中會有異常,那么countDownLatch.countDown()應該寫在finally里面,這樣才能保證異常后也能對計數器減1,不會讓主線程永遠等待。
另外,await()方法還有一個實用的重載方法:public booleanawait(long timeout, TimeUnit unit),設置超時時間。
例如上面的代碼,想要設置超時時間10秒,到了10秒無論是否倒數完成到0,都會不再阻塞主線程。返回值是boolean類型,如果是超時返回false,如果計數到達0沒有超時返回true。

// 設置超時時間為10秒 boolean timeoutFlag = countDownLatch.await(10,TimeUnit.SECONDS); if(timeoutFlag) { System.out.println("所有子線程執行完成"); } else { System.out.println("超時"); }

4、主線程等待線程池

Java線程池java.util.concurrent.ExecutorService是很好用的多線程管理方式。ExecutorService的一個方法boolean awaitTermination(long timeout, TimeUnit unit),即阻塞主線程,等待線程池的所有線程執行完成,用法和上面所說的CountDownLatch的public boolean await(long timeout,TimeUnit unit)類似,參數設置一個超時時間,返回值是boolean類型,如果超時返回false,如果線程池中的線程全部執行完成,返回true。
由于ExecutorService沒有類似CountDownLatch的無參數的await()方法,只能通過awaitTermination來實現主線程等待線程池。

public class Main { public static void main(String[] args) { long start = System.currentTimeMillis(); // 創建一個同時允許兩個線程并發執行的線程池 ExecutorService executor = Executors.newFixedThreadPool(2); for(int i = 0; i < 5; i++) { Thread thread = new TestThread(); executor.execute(thread); } executor.shutdown(); try { // awaitTermination返回false即超時會繼續循環,返回true即線程池中的線程執行完成主線程跳出循環往下執行,每隔10秒循環一次 while (!executor.awaitTermination(10, TimeUnit.SECONDS)); } catch (InterruptedException e) { e.printStackTrace(); } long end = System.currentTimeMillis(); System.out.println("子線程執行時長:" + (end - start)); } }

執行結果:

Thread-0子線程開始 Thread-1子線程開始 Thread-0子線程結束 Thread-2子線程開始 Thread-1子線程結束 Thread-3子線程開始 Thread-2子線程結束 Thread-4子線程開始 Thread-3子線程結束 Thread-4子線程結束 子線程執行時長:15000

另外,while(!executor.isTerminated())也可以替代上面的while (!executor.awaitTermination(10,TimeUnit.SECONDS)),isTerminated()是用來判斷線程池是否執行完成。但是二者比較我認為還是awaitTermination更好,它有一個超時時間可以控制每隔多久循環一次,而不是一直在循環來消耗性能。

總結

以上是生活随笔為你收集整理的Java主线程等待子线程、线程池的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。