激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務(wù)器之家:專(zhuān)注于服務(wù)器技術(shù)及軟件下載分享
分類(lèi)導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|JavaScript|易語(yǔ)言|

服務(wù)器之家 - 編程語(yǔ)言 - Java教程 - JVM中的守護(hù)線程示例詳解

JVM中的守護(hù)線程示例詳解

2021-06-28 10:23技術(shù)小黑屋 Java教程

這篇文章主要給大家介紹了關(guān)于JVM中守護(hù)線程的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

前言

在java中有兩類(lèi)線程:user thread(用戶(hù)線程)、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è)很稱(chēng)職的守護(hù)者。

在之前的《詳解jvm如何處理異常》提到了守護(hù)線程,當(dāng)時(shí)沒(méi)有詳細(xì)解釋?zhuān)源蛩惴诺浇裉靵?lái)解釋說(shuō)明一下jvm守護(hù)線程的內(nèi)容。

特點(diǎn)

  • 通常由jvm啟動(dòng)
  • 運(yùn)行在后臺(tái)處理任務(wù),比如垃圾回收等
  • 用戶(hù)啟動(dòng)線程執(zhí)行結(jié)束或者jvm結(jié)束時(shí),會(huì)等待所有的非守護(hù)線程執(zhí)行結(jié)束,但是不會(huì)因?yàn)槭刈o(hù)線程的存在而影響關(guān)閉。

判斷線程是否為守護(hù)線程

判斷一個(gè)線程是否為守護(hù)線程,主要依據(jù)如下的內(nèi)容

?
1
2
3
4
5
6
7
8
9
10
11
12
13
/* 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í)。

?
1
2
3
4
5
6
7
8
9
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());
 }
}

線程睡眠的方法

?
1
2
3
4
5
6
7
8
private static void makethreadsleep(long durationinmillseconds) {
 try {
 thread.sleep(durationinmillseconds);
 } catch (interruptedexception e) {
 e.printstacktrace();
 }
 
}

驗(yàn)證普通的(非守護(hù)線程)線程會(huì)影響進(jìn)程(jvm)退出

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
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不等待用戶(hù)啟動(dòng)的守護(hù)線程結(jié)束就退出的事實(shí)。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
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 = 3006

process finished with exit code 0

我們可以看到,上面的日志沒(méi)有類(lè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ì)服務(wù)器之家的支持。

原文鏈接:https://droidyue.com/blog/2018/12/16/daemon-thread-in-java/

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 男人久久天堂 | 污在线观看网站 | av电影在线免费 | 国产一区二区亚洲 | 亚洲一区二区三区精品在线观看 | 免费视频aaa| 深夜影院一级毛片 | 神秘电影91 | 在线观看国产www | 国产中出视频 | 色奇米 | 欧美日韩免费看 | 成人一区二区在线观看视频 | 精选久久| 精品国产91久久久久久浪潮蜜月 | 欧美性生活久久久 | 中文字幕 日本 | 国产日本在线播放 | 黄片毛片一级 | 久久毛片免费观看 | 天使萌一区二区三区免费观看 | 2021国产精品 | 欧美成年性h版影视中文字幕 | 手机视频在线播放 | 黄色毛片前黄 | 久久99精品久久久久久秒播蜜臀 | 成人做爽爽爽爽免费国产软件 | 99爱精品在线| 国产精品久久久久久久久久尿 | www.国产一区.com | 黄色成人短视频 | 黄视频网站免费观看 | 日本精品免费观看 | 国产亚洲精品久久久久久久久久 | 久久久久久久久久久亚洲 | 日本中文字幕网址 | 精品在线观看一区二区 | 国产一区二区三区视频免费 | 久草在线综合网 | 一类黄色大片 | 国产精品久久久久久久久久iiiii |