談?wù)凧ava中的守護線程與普通線程
守護線程與普通線程的唯一區(qū)別是:當(dāng)JVM中所有的線程都是守護線程的時候,JVM就可以退出了;如果還有一個或以上的非守護線程則不會退出。(以上是針對正常退出,調(diào)用System.exit則必定會退出)
所以setDeamon(true)的唯一意義就是告訴JVM不需要等待它退出,讓JVM喜歡什么退出就退出吧,不用管它。
守護線程在沒有用戶線程可服務(wù)時自動離開,在Java中比較特殊的線程是被稱為守護(Daemon)線程的低級別線程。這個線程具有最低的優(yōu)先級,用于為系統(tǒng)中的其它對象和線程提供服務(wù)。將一個用戶線程設(shè)置為守護線程的方式是在線程對象創(chuàng)建之前調(diào)用線程對象的setDaemon方法。典型的守護線程例子是JVM中的系統(tǒng)資源自動回收線程,我們所熟悉的Java垃圾回收線程就是一個典型的守護線程,當(dāng)我們的程序中不再有任何運行中的Thread,程序就不會再產(chǎn)生垃圾,垃圾回收器也就無事可做,所以當(dāng)垃圾回收線程是Java虛擬機上僅剩的線程時,Java虛擬機會自動離開。它始終在低級別的狀態(tài)中運行,用于實時監(jiān)控和管理系統(tǒng)中的可回收資源。守護進程(Daemon)是運行在后臺的一種特殊進程。它獨立于控制終端并且周期性地執(zhí)行某種任務(wù)或等待處理某些發(fā)生的事件。也就是說守護線程不依賴于終端,但是依賴于系統(tǒng),與系統(tǒng)“同生共死”。那Java的守護線程是什么樣子的呢。當(dāng)JVM中所有的線程都是守護線程的時候,JVM就可以退出了;如果還有一個或以上的非守護線程則JVM不會退出。
Java有兩種Thread:“守護線程Daemon”與“用戶線程User”。
我們之前看到的例子都是用戶,守護線程是一種“在后臺提供通用性支持”的線程,它并不屬于程序本體。
從字面上我們很容易將守護線程理解成是由虛擬機(virtual machine)在內(nèi)部創(chuàng)建的,而用戶線程則是自己所創(chuàng)建的。事實并不是這樣,任何線程都可以是“守護線程Daemon”或“用戶線程User”。他們在幾乎每個方面都是相同的,唯一的區(qū)別是判斷虛擬機何時離開:
- 用戶線程:Java虛擬機在它所有非守護線程已經(jīng)離開后自動離開。
- 守護線程:守護線程則是用來服務(wù)用戶線程的,如果沒有其他用戶線程在運行,那么就沒有可服務(wù)對象,也就沒有理由繼續(xù)下去。
setDaemon(boolean on)方法可以方便的設(shè)置線程的Daemon模式,true為Daemon模式,false為User模式。setDaemon(boolean on)方法必須在線程啟動之前調(diào)用,當(dāng)線程正在運行時調(diào)用會產(chǎn)生異常。isDaemon方法將測試該線程是否為守護線程。值得一提的是,當(dāng)你在一個守護線程中產(chǎn)生了其他線程,那么這些新產(chǎn)生的線程不用設(shè)置Daemon屬性,都將是守護線程,用戶線程同樣。
package ying.thread; import java.io.IOException; public class Test extends Thread { public Test() { } public void run() { for (int i = 0 ; i < 100 ; i ++) { try { Thread.sleep(100) ; } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } System.out.println(i); } } public static void main (String args[]) { Test test = new Test() ; test.setDaemon(true) ; test.start() ; System.out.println("isDaemon=" + test.isDaemon()); try { System.in.read() ; } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
如果沒有用戶線程,那么守護線程也沒有存活下去的意義了:
package ying.thread; import java.io.IOException; public class Test extends Thread { public Test() { } public void run() { for (int i = 0 ; i < 100 ; i ++) { try { Thread.sleep(100) ; } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } System.out.println(i); } } public static void main (String args[]) { Test test = new Test() ; test.setDaemon(true) ; test.start() ; System.out.println("isDaemon=" + test.isDaemon()); } }
這個程序執(zhí)行之后什么也不執(zhí)行;什么也不打??;
如果我們把 thread.setDaemon(true);刪除,那么就可以打印出數(shù)字了。
這就是守護線程,守護著最后一個用戶線程,如果沒有用戶線程了,他也沒作用了。不退出等什么??
以上就是談?wù)凧ava中的守護線程與普通線程的詳細內(nèi)容,更多關(guān)于Java 守護線程與普通線程的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
詳解Spring Boot使用redis實現(xiàn)數(shù)據(jù)緩存
本篇文章主要介紹了詳解Spring Boot使用redis實現(xiàn)數(shù)據(jù)緩存,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-04-04Springboot如何獲取配置文件application.yml中自定義的變量并使用
這篇文章主要介紹了Springboot中獲取配置文件(application.yml)中自定義的變量并使用,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-09-09Java中BM(Boyer-Moore)算法的圖解與實現(xiàn)
本文主要介紹了兩個大的部分,第一部分通過圖解的方式講解BM算法,第二部分則代碼實現(xiàn)一個簡易的BM算法,感興趣的小伙伴可以學(xué)習(xí)一下2022-05-05springboot+redis實現(xiàn)微博熱搜排行榜的示例代碼
本文主要介紹了springboot+redis實現(xiàn)微博熱搜排行榜的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05java編程實現(xiàn)獲取服務(wù)器IP地址及MAC地址的方法
這篇文章主要介紹了java編程實現(xiàn)獲取機器IP地址及MAC地址的方法,實例分析了Java分別針對單網(wǎng)卡及多網(wǎng)卡的情況下獲取服務(wù)器IP地址與MAC地址的相關(guān)技巧,需要的朋友可以參考下2015-11-11Springboot基于maven打包分離lib及resource
這篇文章主要介紹了Springboot基于maven打包分離lib及resource,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-10-10