JVM中的守護(hù)線程示例詳解
前言
在Java中有兩類線程:User Thread(用戶線程)、Daemon Thread(守護(hù)線程)
用個(gè)比較通俗的比如,任何一個(gè)守護(hù)線程都是整個(gè)JVM中所有非守護(hù)線程的保姆:
只要當(dāng)前JVM實(shí)例中尚存在任何一個(gè)非守護(hù)線程沒(méi)有結(jié)束,守護(hù)線程就全部工作;只有當(dāng)最后一個(gè)非守護(hù)線程結(jié)束時(shí),守護(hù)線程隨著JVM一同結(jié)束工作。
Daemon的作用是為其他線程的運(yùn)行提供便利服務(wù),守護(hù)線程最典型的應(yīng)用就是 GC (垃圾回收器),它就是一個(gè)很稱職的守護(hù)者。
在之前的《詳解JVM如何處理異常》提到了守護(hù)線程,當(dāng)時(shí)沒(méi)有詳細(xì)解釋,所以打算放到今天來(lái)解釋說(shuō)明一下JVM守護(hù)線程的內(nèi)容。
特點(diǎn)
- 通常由JVM啟動(dòng)
- 運(yùn)行在后臺(tái)處理任務(wù),比如垃圾回收等
- 用戶啟動(dòng)線程執(zhí)行結(jié)束或者JVM結(jié)束時(shí),會(huì)等待所有的非守護(hù)線程執(zhí)行結(jié)束,但是不會(huì)因?yàn)槭刈o(hù)線程的存在而影響關(guān)閉。
判斷線程是否為守護(hù)線程
判斷一個(gè)線程是否為守護(hù)線程,主要依據(jù)如下的內(nèi)容
/* Whether or not the thread is a daemon thread. */
private boolean daemon = false;
/**
* Tests if this thread is a daemon thread.
*
* @return <code>true</code> if this thread is a daemon thread;
* <code>false</code> otherwise.
* @see #setDaemon(boolean)
*/
public final boolean isDaemon() {
return daemon;
}
下面我們進(jìn)行一些簡(jiǎn)單的代碼,驗(yàn)證一些關(guān)于守護(hù)線程的特性和一些猜測(cè)。
輔助方法
打印線程信息的方法,輸出線程的組,是否為守護(hù)線程以及對(duì)應(yīng)的優(yōu)先級(jí)。
private static void dumpAllThreadsInfo() {
Set<Thread> threadSet = Thread.getAllStackTraces().keySet();
for(Thread thread: threadSet) {
System.out.println("dumpAllThreadsInfo thread.name=" + thread.getName()
+ ";group=" + thread.getThreadGroup()
+ ";isDaemon=" + thread.isDaemon()
+ ";priority=" + thread.getPriority());
}
}
線程睡眠的方法
private static void makeThreadSleep(long durationInMillSeconds) {
try {
Thread.sleep(durationInMillSeconds);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
驗(yàn)證普通的(非守護(hù)線程)線程會(huì)影響進(jìn)程(JVM)退出
private static void testNormalThread() {
long startTime = System.currentTimeMillis();
new Thread("NormalThread") {
@Override
public void run() {
super.run();
//保持睡眠,確保在執(zhí)行dumpAllThreadsInfo時(shí),該線程不會(huì)因?yàn)橥顺鰧?dǎo)致dumpAllThreadsInfo無(wú)法打印信息。
makeThreadSleep(10 * 1000);
System.out.println("startNormalThread normalThread.time cost=" + (System.currentTimeMillis() - startTime));
}
}.start();
//主線程暫定3秒,確保子線程都啟動(dòng)完成
makeThreadSleep(3 * 1000);
dumpAllThreadsInfo();
System.out.println("MainThread.time cost = " + (System.currentTimeMillis() - startTime));
}
獲取輸出日志
dumpAllThreadsInfo thread.name=Signal Dispatcher;group=java.lang.ThreadGroup[name=system,maxpri=10];isDaemon=true;priority=9
dumpAllThreadsInfo thread.name=Attach Listener;group=java.lang.ThreadGroup[name=system,maxpri=10];isDaemon=true;priority=9
dumpAllThreadsInfo thread.name=Monitor Ctrl-Break;group=java.lang.ThreadGroup[name=main,maxpri=10];isDaemon=true;priority=5
dumpAllThreadsInfo thread.name=Reference Handler;group=java.lang.ThreadGroup[name=system,maxpri=10];isDaemon=true;priority=10
dumpAllThreadsInfo thread.name=main;group=java.lang.ThreadGroup[name=main,maxpri=10];isDaemon=false;priority=5
dumpAllThreadsInfo thread.name=NormalThread;group=java.lang.ThreadGroup[name=main,maxpri=10];isDaemon=false;priority=5
dumpAllThreadsInfo thread.name=Finalizer;group=java.lang.ThreadGroup[name=system,maxpri=10];isDaemon=true;priority=8
MainThread.time cost = 3009
startNormalThread normalThread.time cost=10003
Process finished with exit code 0 結(jié)束進(jìn)程
我們根據(jù)上面的日志,我們可以發(fā)現(xiàn)
startNormalThread normalThread.time cost=10003代表著子線程執(zhí)行結(jié)束,先于后面的進(jìn)程結(jié)束執(zhí)行。Process finished with exit code 0代表 結(jié)束進(jìn)程
以上日志可以驗(yàn)證進(jìn)程是在我們啟動(dòng)的子線程結(jié)束之后才退出的。
驗(yàn)證JVM不等待守護(hù)線程就會(huì)結(jié)束
其實(shí)上面的例子也可以驗(yàn)證JVM不等待JVM啟動(dòng)的守護(hù)線程(Reference Handler,Signal Dispatcher等)執(zhí)行結(jié)束就退出。
這里我們?cè)俅斡靡欢未a驗(yàn)證一下JVM不等待用戶啟動(dòng)的守護(hù)線程結(jié)束就退出的事實(shí)。
private static void testDaemonThread() {
long startTime = System.currentTimeMillis();
Thread daemonThreadSetByUser = new Thread("daemonThreadSetByUser") {
@Override
public void run() {
makeThreadSleep(10 * 1000);
super.run();
System.out.println("daemonThreadSetByUser.time cost=" + (System.currentTimeMillis() - startTime));
}
};
daemonThreadSetByUser.setDaemon(true);
daemonThreadSetByUser.start();
//主線程暫定3秒,確保子線程都啟動(dòng)完成
makeThreadSleep(3 * 1000);
dumpAllThreadsInfo();
System.out.println("MainThread.time cost = " + (System.currentTimeMillis() - startTime));
}
上面的結(jié)果得到的輸出日志為
dumpAllThreadsInfo thread.name=Signal Dispatcher;group=java.lang.ThreadGroup[name=system,maxpri=10];isDaemon=true;priority=9
dumpAllThreadsInfo thread.name=Attach Listener;group=java.lang.ThreadGroup[name=system,maxpri=10];isDaemon=true;priority=9
dumpAllThreadsInfo thread.name=Monitor Ctrl-Break;group=java.lang.ThreadGroup[name=main,maxpri=10];isDaemon=true;priority=5
dumpAllThreadsInfo thread.name=Reference Handler;group=java.lang.ThreadGroup[name=system,maxpri=10];isDaemon=true;priority=10
dumpAllThreadsInfo thread.name=main;group=java.lang.ThreadGroup[name=main,maxpri=10];isDaemon=false;priority=5
dumpAllThreadsInfo thread.name=daemonThreadSetByUser;group=java.lang.ThreadGroup[name=main,maxpri=10];isDaemon=true;priority=5
dumpAllThreadsInfo thread.name=Finalizer;group=java.lang.ThreadGroup[name=system,maxpri=10];isDaemon=true;priority=8
MainThread.time cost = 3006Process finished with exit code 0
我們可以看到,上面的日志沒(méi)有類似daemonThreadSetByUser.time cost=的信息??梢源_定JVM沒(méi)有等待守護(hù)線程結(jié)束就退出了。
注意:
- 新的線程是否初始為守護(hù)線程,取決于啟動(dòng)該線程的線程是否為守護(hù)線程。
- 守護(hù)線程默認(rèn)啟動(dòng)的線程為守護(hù)線程,非守護(hù)線程啟動(dòng)的線程默認(rèn)為非守護(hù)線程。
- 主線程(非守護(hù)線程)啟用一個(gè)守護(hù)線程,需要調(diào)用Thread.setDaemon來(lái)設(shè)置啟動(dòng)線程為守護(hù)線程。
關(guān)于Priority與守護(hù)線程的關(guān)系
有一種傳言為守護(hù)線程的優(yōu)先級(jí)要低,然而事實(shí)是
- 優(yōu)先級(jí)與是否為守護(hù)線程沒(méi)有必然的聯(lián)系
- 新的線程的優(yōu)先級(jí)與創(chuàng)建該線程的線程優(yōu)先級(jí)一致。
- 但是建議將守護(hù)線程的優(yōu)先級(jí)降低一些。
感興趣的可以自己驗(yàn)證一下(其實(shí)上面的代碼已經(jīng)有驗(yàn)證了)
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
Java Spring Boot實(shí)戰(zhàn)練習(xí)之單元測(cè)試篇
單元測(cè)試(unit testing),是指對(duì)軟件中的最小可測(cè)試單元進(jìn)行檢查和驗(yàn)證。對(duì)于單元測(cè)試中單元的含義,一般來(lái)說(shuō),要根據(jù)實(shí)際情況去判定其具體含義,如C語(yǔ)言中單元指一個(gè)函數(shù),Java里單元指一個(gè)類,圖形化的軟件中可以指一個(gè)窗口或一個(gè)菜單等2021-10-10
ThreadPoolExecutor參數(shù)含義及源碼執(zhí)行流程詳解
這篇文章主要為大家介紹了ThreadPoolExecutor參數(shù)含義及源碼執(zhí)行流程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11
Java基礎(chǔ)之ArrayList的擴(kuò)容機(jī)制
這篇文章主要介紹了Java基礎(chǔ)之ArrayList的擴(kuò)容機(jī)制,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java基礎(chǔ)的小伙伴們有很好的幫助,需要的朋友可以參考下2021-05-05
Java中數(shù)據(jù)轉(zhuǎn)換及字符串的“+”操作方法
本文主要介紹了Java中的數(shù)據(jù)類型轉(zhuǎn)換,包括隱式轉(zhuǎn)換和強(qiáng)制轉(zhuǎn)換,隱式轉(zhuǎn)換通常用于將范圍較小的數(shù)據(jù)類型轉(zhuǎn)換為范圍較大的數(shù)據(jù)類型,而強(qiáng)制轉(zhuǎn)換則是將范圍較大的數(shù)據(jù)類型轉(zhuǎn)換為范圍較小的數(shù)據(jù)類型,本文介紹Java中數(shù)據(jù)轉(zhuǎn)換以及字符串的“+”操作,感興趣的朋友一起看看吧2024-10-10
SpringBoot的@Value注解如何設(shè)置默認(rèn)值
這篇文章主要介紹了SpringBoot的@Value注解如何設(shè)置默認(rèn)值問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-02-02
Spring AOP注解失效的坑及JDK動(dòng)態(tài)代理
這篇文章主要介紹了Spring AOP注解失效的坑及JDK動(dòng)態(tài)代理,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03
springboot應(yīng)用訪問(wèn)zookeeper的流程
這篇文章主要介紹了springboot應(yīng)用訪問(wèn)zookeeper的流程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01

