Java中的守護線程問題
守護線程
在Java中有兩類線程
- User Thread(用戶線程)
- Daemon Thread(守護線程)
守護線程的功能非常簡單,在其本身是一個線程的同時,主要是為了給其他的線程提供服務(wù),比如說計時器,清空高速緩存等等操作,守護線程具有和被守護線程一樣的生命周期(這里并不是說守護線程和被守護線程常常是1-1的關(guān)系),當(dāng)被守護線程死亡,守護線程往往也會死亡,當(dāng)虛擬機中只剩下守護線程時,虛擬機就會退出,因為此時也沒有運行程序的必要了
一個比較通俗的解釋:任何一個守護線程都是整個JVM中所有非守護線程的保姆
只要當(dāng)前JVM實例中尚存在任何一個非守護線程沒有結(jié)束,守護線程就全部工作;只有當(dāng)最后一個非守護線程結(jié)束時,守護線程隨著JVM一同結(jié)束工作。
守護線程的作用是為其他線程的運行提供便利服務(wù),守護線程最典型的應(yīng)用就是 GC (垃圾回收器),它就是一個很稱職的守護者。
需要注意的點是:
- 守護線程的優(yōu)先級比較低
- 守護線程要注意考慮關(guān)機動作
- 守護線程應(yīng)該永遠不去訪問固有資源,比如說文件或者數(shù)據(jù)庫,因為它會在任何時候甚至一個操作的中間發(fā)生中斷。
不要給守護線程分擔(dān)讀寫邏輯或者計算邏輯,因為無法確定守護線程是否已經(jīng)完成了工作,但是只要User退出守護線程也會立馬結(jié)束,對于計算機程序來說這樣的程序可能多次運行結(jié)果不一樣,很顯然這對于程序來說是毀滅性的。
操作:
通過thread.setDaemon(true) 將線程轉(zhuǎn)換為守護線程
這個方法必須在thread.start()之前進行調(diào)用
守護線程與用戶線程的區(qū)別
介紹Java守護線程與用戶線程的概念和使用方法,以及相關(guān)注意事項。
1 定義和概述
Java 中的線程分為兩類,分別為daemon 線程(守護線程〉和user 線程(用戶線程)。守護線程又稱Daemon線程,運行在后臺,看不見;用戶線程運行在前臺,看的見。
在JVM啟動時會調(diào)用main 函數(shù), main 函數(shù)所在的線程就是一個用戶線程,其實在JVM內(nèi)部同時-還啟動了好多守護線程, 比如垃圾回收線程。
Daemon線程是一種支持型線程,因為它主要被用作程序中后臺調(diào)度以及支持性工作。這意味著,當(dāng)一個Java虛擬機中不存在非Daemon線程的時候,Java虛擬機將會退出,而不管當(dāng)前是否有守護線程,也就是說守護線程是否結(jié)束并不影響JVM的退出。
實際上,在main線程運行結(jié)束后,JVM會自動啟動一個叫作DestroyJavaVM 的線程,該線程會等待所有用戶線程結(jié)束后終止JVM 進程。
在Tomcat的NIO實現(xiàn)NioEndpoint中會開啟一組接受線程來接受用戶的連接請求,以及一組處理線程負責(zé)具體處理用戶請求,在默認情況下,接受線程和處理線程都是守護線程,這意味著當(dāng)tomcat 收到shutdown 命令后并且沒有其他用戶線程存在的情況下tomcat 進程會馬上消亡,而不會等待處理線程處理完當(dāng)前的請求。
2 使用守護線程
在線程start之前,可以通過調(diào)用thread.setDaemon(true)將線程設(shè)置為Daemon線程。
守護線程有兩種結(jié)束方式:
- 守護線程也具有自己的run();方法,當(dāng)后臺線程完成自己的run方法后,守護線程結(jié)束。
- 用戶線程運行結(jié)束,守護線程自動結(jié)束。
3 測試案例
public class Daemon { //啟動該類,將會構(gòu)造兩條線程,main線程和一條子線程。 public static void main(String[] args) throws InterruptedException { //測試非守護線程 //可以看到,輸出"main線程結(jié)束"之后,子線程還在繼續(xù)輸出,程序沒有結(jié)束 // test1(); //測試守護線程 //可以看到,輸出"main線程結(jié)束"之后,子線程沒有繼續(xù)輸出,程序結(jié)束 test2(); } /** * 測試非守護線程 * * @throws InterruptedException */ public static void test1() throws InterruptedException { Thread thread = new Thread(() -> { while (true) { try { Thread.currentThread().sleep(500); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("子線程非守護線程"); } }); thread.start(); Thread.currentThread().sleep(1000); System.out.println("main線程結(jié)束"); } //測試守護線程 public static void test2() throws InterruptedException { Thread thread = new Thread(() -> { while (true) { try { Thread.currentThread().sleep(500); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("子線程守護線程"); } }); thread.setDaemon(true); thread.start(); Thread.currentThread().sleep(1000); System.out.println("main線程結(jié)束"); } }
使用jps查看java進程,可以發(fā)現(xiàn)如果子線程是守護線程那么主線程結(jié)束,子線程也結(jié)束了;如果子線程不是守護線程那么主線程結(jié)束,子線程沒有結(jié)束。
4 注意事項
Daemon線程被用作完成支持性工作,但是在Java虛擬機退出時Daemon線程中的finally塊并不一定會執(zhí)行,如下代碼:
public class Daemon { public static void main(String[] args) { Thread thread = new Thread(new DaemonRunner(), "DaemonRunner"); thread.setDaemon(true); thread.start(); } static class DaemonRunner implements Runnable { @Override public void run() { try { SleepUtils.second(10); } finally { System.out.println("DaemonThread finally run."); } } } }
運行Daemon程序,可以看到在控制臺上沒有任何輸出。main線程(非Daemon線程)在啟動了線程DaemonRunner之后隨著main方法執(zhí)行完畢而終止,而此時Java虛擬機中已經(jīng)沒有非Daemon線程,虛擬機需要退出。Java虛擬機中的所有Daemon線程都需要立即終止,因此DaemonRunner立即終止,但是DaemonRunner中的finally塊并沒有執(zhí)行。
在構(gòu)建Daemon線程時,不能依靠finally塊中的內(nèi)容來確保執(zhí)行關(guān)閉或清理資源的邏輯。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Eclipse下編寫java程序突然不會自動生成R.java文件和包的解決辦法
這篇文章主要介紹了Eclipse下編寫java程序突然不會自動生成R.java文件和包的解決辦法 的相關(guān)資料,需要的朋友可以參考下2016-01-01解決Maven本地倉庫明明有對應(yīng)的jar包但還是報找不到的問題
這篇文章主要介紹了解決Maven本地倉庫明明有對應(yīng)的jar包但還是報找不到的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-10-10帶你了解Java數(shù)據(jù)結(jié)構(gòu)和算法之無權(quán)無向圖
這篇文章主要為大家介紹了Java數(shù)據(jù)結(jié)構(gòu)和算法之無權(quán)無向圖?,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-01-01Java?Stream函數(shù)式編程管道流結(jié)果處理
這篇文章主要為大家介紹了Java?Stream函數(shù)式編程管道流結(jié)果處理的示例過程解析需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步2022-03-03