Java中的守護線程 & 非守護線程
守護線程 (Daemon Thread)
非守護線程,又稱用戶線程(User Thread)
用個比較通俗的好比,任何一個守護線程都是整個JVM中全部非守護線程的保姆:只要當前JVM實例中尚存在任何一個非守護線程沒有結束,守護線程就所有工做;只有當最後一個非守護線程結束時,守護線程隨着JVM一同結束工做。
守護線程最典型的應用就是 GC (垃圾回收器)
守護線程一般是由虛擬機自行建立使用,不過經過編碼一樣能夠建立守護線程,並且 very easy:
Thread daemonTread = new Thread();
// 設定 daemonThread 爲 守護線程,default false(非守護線程)
daemonThread.setDaemon(true);
// 驗證當前線程是否爲守護線程,返回 true 則爲守護線程
daemonThread.isDaemon();
守護線程與普通線程的惟一區別是:當JVM中全部的線程都是守護線程的時候,JVM就能夠退出了;若是還有一個或以上的非守護線程則不會退出。(以上是針對正常退出,調用System.exit則一定會退出)
因此setDeamon(true)的惟一意義就是告訴JVM不須要等待它退出,讓JVM喜歡什麼退出就退出吧,不用管它。數據庫
//下面是一個例子編碼
守護線程與普通線程寫法上基本麼啥區別,調用線程對象的方法setDaemon(true),則能夠將其設置爲守護線程。
守護線程使用的狀況較少,但並不是無用,舉例來講,JVM的垃圾回收、內存管理等線程都是守護線程。還有就是在作數據庫應用時候,使用的數據庫鏈接池,鏈接池自己也包含着不少後臺線程,監控鏈接個數、超時時間、狀態等等。
setDaemon方法的詳細說明:
public
final
void setDaemon(
boolean on)將該線程標記爲守護線程或用戶線程。當正在運行的線程都是守護線程時,Java 虛擬機退出。
該方法必須在啓動線程前調用。
該方法首先調用該線程的 checkAccess 方法,且不帶任何參數。這可能拋出 SecurityException(在當前線程中)。
參數:
on - 若是爲
true,則將該線程標記爲守護線程。
拋出:
IllegalThreadStateException - 若是該線程處於活動狀態。
SecurityException - 若是當前線程沒法修改該線程。
另請參見:
isDaemon(), checkAccess()
/**
* Java線程:線程的調度-守護線程
*
* @author leizhimin 2009-11-4 9:02:40
*/
public
class Test {
public
static
void main(String[] args) {
Thread t1 =
new MyCommon();
Thread t2 =
new Thread(
new MyDaemon());
t2.setDaemon(
true);
//設置爲守護線程
t2.start();
t1.start();
}
}
class MyCommon
extends Thread {
public
void run() {
for (
int i = 0; i < 5; i++) {
System.out.println(
"線程1第" + i +
"次執行!");
try {
Thread.sleep(7);
}
catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
class MyDaemon
implements Runnable {
public
void run() {
for (
long i = 0; i < 9999999L; i++) {
System.out.println(
"後臺線程第" + i +
"次執行!");
try {
Thread.sleep(7);
}
catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
後臺線程第0次執行!
線程1第0次執行!
線程1第1次執行!
後臺線程第1次執行!
後臺線程第2次執行!
線程1第2次執行!
線程1第3次執行!
後臺線程第3次執行!
線程1第4次執行!
後臺線程第4次執行!
後臺線程第5次執行!
後臺線程第6次執行!
後臺線程第7次執行!
Process finished with exit code 0
從上面的執行結果能夠看出:
前臺線程是保證執行完畢的,後臺線程尚未執行完畢就退出了。