干java 開發這么多年, 之前一直沒留意java 進程還區分守護進程和用戶進程。守護進程這個概念最早還是在linux系統中接觸的,直到近期使用java開發心跳檢測功能時,使用timer時才發現原來java也有守護線程的概念。
1. java 線程
1.1 守護線程與用戶線程
java 線程分為守護線程(daemonthread) 和 用戶線程(userthread)兩類.
- 通常情況下,我們使用thread 創建的線程在默認情況下都屬于用戶線程, 當在啟動線程之前, 執行thread.setdaemon(true)時, 線程會變成守護線程。
- 其實在本質上,用戶線程和守護線程并沒有太大區別,唯一的區別就是會影響虛擬機的退出(程序的終止)。當jvm中只剩下守護線程時,虛擬機會退出,及程序終止;而當jvm中至少擁有一個用戶線程時,jvm都不會退出。
1.2 守護線程特點
java 中的守護線程和linux 中的守護進程還是有些區別的, linux 守護進程時系統級別的, 當系統退出時, 才會終止. 而java 中的守護線程時jvm 級別的, 當jvm中無任何用戶進程時, 守護進程銷毀, jvm 退出, 程序終止. 筆者認為java 守護進程的最主要的特點有:
- 守護進程是運行在程序后臺的線程
- 守護進程創建的進程,依然時守護進程
- 守護進程不會影響jvm的退出,當jvm只剩余守護進程時,jvm 進行退出
- 守護進行在jvm退出時,自動銷毀
在開發java 守護線程時, 需要注意:
- 在線程啟動之前, 及執行thread.start() 方法前, 設置thread.setdemon(true);
- 在守護線程中啟動的子線程也是守護線程
- 守護線程不建議進行寫操作, 因為守護進程隨時可能結束。
1.3 守護線程適用場景
針對于守護線程的特點,筆者認為,java 守護線程通常可用于開發一些為其它用戶線程服務的功能。比如說心跳檢測,事件監聽等。java 中最有名的守護進程當屬gc(垃圾回收)
2. java 守護線程開發
開發守護進程時需要注意兩點:
- 必須在線程啟動之前,即執行thread.start()方法前, 執行thread.setdaemon(true) 方法,否則會拋出異常illegalthreadstateexception
- 守護線程中創建的線程也是守護線程
2.1 測試程序退出
- 注釋thread.setdaemon(true)時, 線程為用戶線程, 程序進行無線循環, 程序不終止。
- 設置thread.setdaemon(true)后, 線程變成守護線程, 程序直接終止, 僅輸出一行信息"main thread done"。 因為程序執行完system 語句之后, main 程序作為唯一的一個用戶線程執行結束了, jvm 中只剩下一個守護進程,所以jvm 便退出了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
public class testthread { public static void main(string[] args) { ansytask ansytask = new ansytask(); thread thread = new thread(ansytask); // 設置線程為異步線程 // thread.setdaemon(true); // 啟動線程 thread.start(); system.out.println( "main thread done" ); } } class ansytask implements runnable{ @override public void run() { while ( true ){ system.out.println(localdatetime.now() + "-hello,thread" ); } } } |
2.2 測試守護線程中創建新的線程
測試會發現, 默認情況下, 守護線程創建的子線程依然是守護線程,用戶創建的守護線程依然時用戶線程。也可以在創建子線程時通過setdaemon()方法修改.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
public class testthread { public static void main(string[] args) throws interruptedexception { ansytask ansytask = new ansytask(); thread thread = new thread(ansytask); // 設置線程為異步線程 thread.setdaemon( true ); // 啟動線程 thread.start(); // 給守護線程點兒執行時間 thread.sleep(1000l); } } class ansytask implements runnable{ @override public void run() { thread thread = new thread( "subthread" ); system.out.println(thread.getname() + " is daemon:" + thread.isdaemon()); } } |
以上所述是小編給大家介紹的java守護線程與用戶線程詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對服務器之家網站的支持!