守護線程與普通線程的唯一區別是:當JVM中所有的線程都是守護線程的時候,JVM就可以退出了;如果還有一個或以上的非守護線程則不會退出。(以上是針對正常退出,調用System.exit則必定會退出)
所以setDeamon(true)的唯一意義就是告訴JVM不需要等待它退出,讓JVM喜歡什么退出就退出吧,不用管它。
守護線程在沒有用戶線程可服務時自動離開,在Java中比較特殊的線程是被稱為守護(Daemon)線程的低級別線程。這個線程具有最低的優先級,用于為系統中的其它對象和線程提供服務。將一個用戶線程設置為守護線程的方式是在線程對象創建之前調用線程對象的setDaemon方法。
典型的守護線程例子是JVM中的系統資源自動回收線程,我們所熟悉的Java垃圾回收線程就是一個典型的守護線程,當我們的程序中不再有任何運行中的Thread,程序就不會再產生垃圾,垃圾回收器也就無事可做,所以當垃圾回收線程是Java虛擬機上僅剩的線程時,Java虛擬機會自動離開。它始終在低級別的狀態中運行,用于實時監控和管理系統中的可回收資源。
守護進程(Daemon)是運行在后臺的一種特殊進程。它獨立于控制終端并且周期性地執行某種任務或等待處理某些發生的事件。也就是說守護線程不依賴于終端,但是依賴于系統,與系統“同生共死”。那Java的守護線程是什么樣子的呢。當JVM中所有的線程都是守護線程的時候,JVM就可以退出了;如果還有一個或以上的非守護線程則JVM不會退出。
Java有兩種Thread:“守護線程Daemon”與“用戶線程User”。
我們之前看到的例子都是用戶,守護線程是一種“在后臺提供通用性支持”的線程,它并不屬于程序本體。
從字面上我們很容易將守護線程理解成是由虛擬機(virtual machine)在內部創建的,而用戶線程則是自己所創建的。
事實并不是這樣,任何線程都可以是“守護線程Daemon”或“用戶線程User”。他們在幾乎每個方面都是相同的,唯一的區別是判斷虛擬機何時離開:
用戶線程:Java虛擬機在它所有非守護線程已經離開后自動離開。
守護線程:守護線程則是用來服務用戶線程的,如果沒有其他用戶線程在運行,那么就沒有可服務對象,也就沒有理由繼續下去。
setDaemon(boolean on)方法可以方便的設置線程的Daemon模式,true為Daemon模式,false為User模式。
setDaemon(boolean on)方法必須在線程啟動之前調用,當線程正在運行時調用會產生異常。
isDaemon方法將測試該線程是否為守護線程。值得一提的是,**當你在一個守護線程中產生了其他線程,那么這些新產生的線程不用設置Daemon屬性,都將是守護線程,用戶線程同樣**
~~~
package testThread;
public class TestThread extends Thread{
@Override
public void run() {
for (int i=0; i<10; i++) {
try{
Thread.sleep(1000);
}catch (Exception e){
e.printStackTrace();
}
System.out.println(i);
}
}
public static void main(String[] args) {
TestThread testThread = new TestThread();
testThread.setDaemon(true);
testThread.start();
System.out.println("isDaemon="+testThread.isDaemon());
try {
System.in.read();
}catch (Exception e) {
e.printStackTrace();
}
}
}
~~~
如果沒有用戶線程,那么守護線程也沒有存活下去的意義了:
~~~
package testThread;
public class TestThread extends Thread{
@Override
public void run() {
for (int i=0; i<10; i++) {
try{
Thread.sleep(1000);
}catch (Exception e){
e.printStackTrace();
}
System.out.println(i);
}
}
public static void main(String[] args) {
TestThread testThread = new TestThread();
testThread.setDaemon(true);
testThread.start();
System.out.println("isDaemon="+testThread.isDaemon());
}
}
~~~
這個程序執行之后什么也不執行;什么也不打印;
如果我們把 thread.setDaemon(true);刪除,那么就可以打印出數字了。
這就是守護線程,守護著最后一個用戶線程,如果沒有用戶線程了,他也沒作用了。不退出等什么??
- linux
- 常用命令
- 高級文本命令
- 面試題
- redis
- String
- list
- hash
- set
- sortedSet
- 案例-推薦
- java高級特性
- 多線程
- 實現線程的三種方式
- 同步關鍵詞
- 讀寫鎖
- 鎖的相關概念
- 多線程的join
- 有三個線程T1 T2 T3,保證順序執行
- java五種線程池
- 守護線程與普通線程
- ThreadLocal
- BlockingQueue消息隊列
- JMS
- 反射
- volatile
- jvm
- IO
- nio
- netty
- netty簡介
- 案例一發送字符串
- 案例二發送對象
- 輕量級RPC開發
- 簡介
- spring(IOC/AOP)
- spring初始化順序
- 通過ApplicationContextAware加載Spring上下文
- InitializingBean的作用
- 結論
- 自定義注解
- zk在框架中的應用
- hadoop
- 簡介
- hadoop集群搭建
- hadoop單機安裝
- HDFS簡介
- hdfs基本操作
- hdfs環境搭建
- 常見問題匯總
- hdfs客戶端操作
- mapreduce工作機制
- 案列-單詞統計
- 局部聚合Combiner
- 案列-流量統計(分區,排序,比較)
- 案列-倒排索引
- 案例-共同好友
- 案列-join算法實現
- 案例-求topN(分組)
- 自定義inputFormat
- 自定義outputFormat
- 框架運算全流程
- mapreduce的優化方案
- HA機制
- Hive
- 安裝
- DDL操作
- 創建表
- 修改表
- DML操作
- Load
- insert
- select
- join操作
- 嚴格模式
- 數據類型
- shell參數
- 函數
- 內置運算符
- 內置函數
- 自定義函數
- Transform實現
- 特殊分割符處理
- 案例
- 級聯求和accumulate
- flume
- 簡介
- 安裝
- 常用的組件
- 攔截器
- 案例
- 采集目錄到HDFS
- 采集文件到HDFS
- 多個agent串聯
- 日志采集和匯總
- 自定義攔截器
- 高可用配置
- 使用注意
- sqoop
- 安裝
- 數據導入
- 導入數據到HDFS
- 導入關系表到HIVE
- 導入表數據子集
- 增量導入
- 數據導出
- 作業
- 原理
- azkaban
- 簡介
- 安裝
- 案例
- 簡介
- command類型單一job
- command類型多job工作流flow
- HDFS操作任務
- mapreduce任務
- hive腳本任務
- hbase
- 簡介
- 安裝
- 命令行
- 基本CURD
- 過濾器查詢
- 系統架構
- 物理存儲
- 尋址機制
- 讀寫過程
- Region管理
- master工作機制
- 建表高級屬性
- 與mapreduce結合
- 協處理器
- 點擊流平臺開發
- 簡介
- storm
- 簡介
- 安裝
- 集群啟動及任務過程分析
- 單詞統計
- 并行度
- ACK容錯機制
- ACK簡介