在Java中有兩類(lèi)線程,分別是User Thread(用戶線程)和Daemon Thread(守護(hù)線程) 。
用戶線程很好理解,我們?nèi)粘i_(kāi)發(fā)中編寫(xiě)的業(yè)務(wù)邏輯代碼,運(yùn)行起來(lái)都是一個(gè)個(gè)用戶線程。而守護(hù)線程相對(duì)來(lái)說(shuō)則要特別理解一下。
什么是守護(hù)線程
在操作系統(tǒng)里面是沒(méi)有所謂的守護(hù)線程的概念的,只有守護(hù)進(jìn)程一說(shuō)。但是Java語(yǔ)言機(jī)制是構(gòu)建在JVM的基礎(chǔ)之上的,這一機(jī)制意味著Java平臺(tái)是把操作系統(tǒng)的底層給屏蔽了起來(lái),所以它可以在它自己的虛擬的平臺(tái)里面構(gòu)造出對(duì)自己有利的機(jī)制。而Java語(yǔ)言或者說(shuō)平臺(tái)的設(shè)計(jì)者多多少少是收到Unix操作系統(tǒng)思想的影響,而守護(hù)線程機(jī)制又是對(duì)JVM這樣的平臺(tái)湊合,于是守護(hù)線程應(yīng)運(yùn)而生。
所謂的守護(hù)線程,指的是程序運(yùn)行時(shí)在后臺(tái)提供的一種通用服務(wù)的線程。比如垃圾回收線程就是一個(gè)很稱(chēng)職的守護(hù)者,并且這種線程并不屬于程序中不可或缺的部分。因此,當(dāng)所有的非守護(hù)線程結(jié)束時(shí),程序也就終止了,同時(shí)會(huì)殺死進(jìn)程中的所有守護(hù)線程。反過(guò)來(lái)說(shuō),只要任何非守護(hù)線程還在運(yùn)行,程序就不會(huì)終止。
事實(shí)上,User Thread(用戶線程)和Daemon Thread(守護(hù)線程)從本質(zhì)上來(lái)說(shuō)并沒(méi)有什么區(qū)別,唯一的不同之處就在于虛擬機(jī)的離開(kāi):如果用戶線程已經(jīng)全部退出運(yùn)行了,只剩下守護(hù)線程存在了,虛擬機(jī)也就退出了。 因?yàn)闆](méi)有了被守護(hù)者,守護(hù)線程也就沒(méi)有工作可做了,也就沒(méi)有繼續(xù)運(yùn)行程序的必要了。
守護(hù)線程的使用與注意事項(xiàng)
守護(hù)線程并非只有虛擬機(jī)內(nèi)部可以提供,用戶也可以手動(dòng)將一個(gè)用戶線程設(shè)定/轉(zhuǎn)換為守護(hù)線程。
在Thread類(lèi)中提供了一個(gè)setDaemon(true)方法來(lái)將一個(gè)普通的線程(用戶線程)設(shè)置為守護(hù)線程。
public final void setDaemon(boolean on);
在使用的過(guò)程中,有幾點(diǎn)需要注意:
1.thread.setDaemon(true)必須在thread.start()之前設(shè)置,否則會(huì)拋出一個(gè)IllegalThreadStateException異常。這也就意味著不能把正在運(yùn)行的常規(guī)線程設(shè)置為守護(hù)線程。 這點(diǎn)與操作系統(tǒng)中的守護(hù)進(jìn)程有著明顯的區(qū)別,守護(hù)進(jìn)程是創(chuàng)建后,讓進(jìn)程擺脫原會(huì)話的控制+讓進(jìn)程擺脫原進(jìn)程組的控制+讓進(jìn)程擺脫原控制終端的控制;所以說(shuō)寄托于虛擬機(jī)的語(yǔ)言機(jī)制跟系統(tǒng)級(jí)語(yǔ)言有著本質(zhì)上面的區(qū)別。
2.在Daemon線程中產(chǎn)生的新線程也是Daemon的。關(guān)于這一點(diǎn)又是與操作系統(tǒng)中的守護(hù)進(jìn)程有著本質(zhì)的區(qū)別:守護(hù)進(jìn)程fork()出來(lái)的子進(jìn)程不再是守護(hù)進(jìn)程,盡管它把父進(jìn)程的進(jìn)程相關(guān)信息復(fù)制過(guò)去了,但是子進(jìn)程的進(jìn)程的父進(jìn)程不是init進(jìn)程,所謂的守護(hù)進(jìn)程本質(zhì)上說(shuō)就是,當(dāng)父進(jìn)程掛掉,init就會(huì)收養(yǎng)該進(jìn)程,然后文件0、1和2都是/dev/null,當(dāng)前目錄到/。
3.不是所有的應(yīng)用都可以分配給Daemon線程來(lái)進(jìn)行服務(wù)的,比如讀寫(xiě)操作或者計(jì)算邏輯。因?yàn)檫@種應(yīng)用可能在Daemon Thread還沒(méi)來(lái)得及進(jìn)行操作時(shí),虛擬機(jī)已經(jīng)退出了。這也就意味著,守護(hù)線程應(yīng)該永遠(yuǎn)不去訪問(wèn)固有資源,如文件、數(shù)據(jù)庫(kù),因?yàn)樗鼤?huì)在任何時(shí)候甚至在一個(gè)操作的中間發(fā)生中斷。
下面以一個(gè)完成文件輸出的守護(hù)線程任務(wù)作為例子:
import java.io.*;
class TestRunnable implements Runnable {
public void run(){
try {
Thread.sleep(1000); // 守護(hù)線程阻塞1秒后運(yùn)行
File f = new File("daemon.txt");
FileOutputStream os = new FileOutputStream(f,true);
os.write("daemon".getBytes());
} catch(IOException e1) {
e1.printStackTrace();
} catch(InterruptedException e2) {
e2.printStackTrace();
}
}
}
public class TestDemo2 {
public static void main(String[] args) throws InterruptedException {
Runnable tr = new TestRunnable();
Thread thread = new Thread(tr);
thread.setDaemon(true); // 設(shè)置守護(hù)線程(必須在thread.start()之前)
thread.start(); // 開(kāi)始執(zhí)行分進(jìn)程 }
}
上面這段代碼的運(yùn)行結(jié)果是文件daemon.txt中沒(méi)有daemon字符串。
但是如果把thread.setDaemon(true);這行代碼注釋掉,文件daemon.txt是可以被寫(xiě)入daemon字符串的,因?yàn)檫@個(gè)時(shí)候這個(gè)線程就是普通的用戶線程了。
簡(jiǎn)單理解就是,JRE判斷程序是否執(zhí)行結(jié)束的標(biāo)準(zhǔn)是所有的前臺(tái)線程(用戶線程)執(zhí)行完畢了,而不管后臺(tái)線程(守護(hù)線程)的狀態(tài)。
守護(hù)線程的應(yīng)用場(chǎng)景
前面說(shuō)了那么多,那么Daemon Thread的實(shí)際應(yīng)用在那里呢?舉個(gè)例子,Web服務(wù)器中的Servlet,在容器啟動(dòng)時(shí),后臺(tái)都會(huì)初始化一個(gè)服務(wù)線程,即調(diào)度線程,負(fù)責(zé)處理http請(qǐng)求,然后每個(gè)請(qǐng)求過(guò)來(lái),調(diào)度線程就會(huì)從線程池中取出一個(gè)工作者線程來(lái)處理該請(qǐng)求,從而實(shí)現(xiàn)并發(fā)控制的目的。也就是說(shuō),一個(gè)實(shí)際應(yīng)用在Java的線程池中的調(diào)度線程。
總結(jié)
從我的理解,守護(hù)線程就是用來(lái)告訴JVM,我的這個(gè)線程是一個(gè)低級(jí)別的線程,不需要等待它運(yùn)行完才退出,讓JVM喜歡什么時(shí)候退出就退出,不用管這個(gè)線程。
在日常的業(yè)務(wù)相關(guān)的CRUD開(kāi)發(fā)中,其實(shí)并不會(huì)關(guān)注到守護(hù)線程這個(gè)概念,也幾乎不會(huì)用上。
但是如果要往更高的地方走的話,這些深層次的概念還是要了解一下的,比如一些框架的底層實(shí)現(xiàn)。
更多關(guān)于“java培訓(xùn)”的問(wèn)題,歡迎咨詢(xún)千鋒教育在線名師。千鋒教育多年辦學(xué),課程大綱緊跟企業(yè)需求,更科學(xué)更嚴(yán)謹(jǐn),每年培養(yǎng)泛IT人才近2萬(wàn)人。不論你是零基礎(chǔ)還是想提升,都可以找到適合的班型,千鋒教育隨時(shí)歡迎你來(lái)試聽(tīng)。