其實,這個問題也可以換個說法:main線程是最後退出的線程嗎?

01 理論分析

當你啟動一個Java Application的時候,這個時候系統創建一個進程,分配各項資源,然後這個進程啟動了Main線程.

我們一般把Main線程說成主線程,因為其他線程一般是由main線程啟動的.但其實,在進程層面看起來,main其實也是一個普通的線程.只不過一些其他的線程都是由main啟動的.

我們可以猜想一下:正常情況下,main線程啟動了其他線程,他們各自執行,彼此不受影響.猜想依據:因為操作系統分配資源的單位是進程,就算main線程退出了,進程也還在.資源還在.在進程看來,線程應該都是平級的,沒有父子關係.這個問題也可以換個說法,main線程是最後退出的線程嗎?

02 實踐驗證

我們模擬一個線程池的例子,從main線程啟動一個線程池,當發生異常,則讓main線程拋出異常終止,看看線程池是否還繼續運行.

這裡由於網上有些爭論認為列印的方式無法判斷main線程是否終止,所以我們的示例使用異常終止,並且使用jconsole工具進行驗證.

public class ThreadPoolException {
ExecutorService threadPool = Executors.newFixedThreadPool(5);
public static void main(String[] args) {
ThreadPoolException t = new ThreadPoolException();
t.futureGet();
}
void futureGet() {
for (int i = 0; i < 5; i++) {
Future future = threadPool.submit(() -> {
System.out.println("current thread name" + Thread.currentThread().getName());
Object object = null;
System.out.print("result## " + object.toString());
});
try {
future.get();
} catch (Exception e) {
System.out.println(Thread.currentThread().getName() + "異常");
// 讓主線程多等一段時間,便於觀察.
try {
Thread.sleep(10000);
} catch (InterruptedException e1) {
e1.printStackTrace();
}
//主線程終止
throw new RuntimeException(Thread.currentThread().getName() + "異常");
}
}
}
}

歡迎大家關注我,需要更多Java面試資料和學習乾貨可以關注我的專欄【Tom貓的Java屋】

Tom貓的java屋?

zhuanlan.zhihu.com圖標

我們可以在jconsole中觀察到:

在main線程先出現然後會消失,線程池中的線程還在,只不過是wait狀態. 在idea中直接運行代碼,也可以觀察到同樣的結論.

03 總結

  • JVM會在所有的非守護線程(用戶線程)執行完畢後退出;
  • main線程是用戶線程;
  • 僅有main線程一個用戶線程執行完畢,不能決定JVM是否退出,也即是說main線程並不一定是最後一個退出的線程。

如果你希望在main線程退出後,所有其他線程也退出.那麼你可以把其他線程都設置為守護線程,也就是setDaemon(true). 對於線程池,你可以在main線程退出的的時候手動進行一些處理.比如shutdown等方法.

04 讀者小福利

在這裡小編整理了很多Java相關的學習資料給大家,還有面試資料和架構資料!有需要的小夥伴們可以在後台私信【java】免費獲取,希望對大家可以有幫助。

Tom貓的java屋?

zhuanlan.zhihu.com
圖標

推薦閱讀:
相关文章