在Java中有两类线程:User Thread(用户线程)、Daemon Thread(守护线程) 可以这样说:任何一个守护线程都是整个JVM中所有非守护线程的保姆:
只要当前JVM实例中尚存在任何一个非守护线程没有结束,守护线程就全部工作;只有当最后一个非守护线程结束时,守护线程随着JVM一同结束工作。
Daemon的作用:为其他线程的运行提供便利服务,守护线程最典型的应用就是 GC (垃圾回收器),它就是一个很称职的守护者。
User和Daemon两者几乎没有区别,唯一的不同之处就在于虚拟机的离开:如果 User Thread已经全部退出运行了,只剩下Daemon Thread存在了,虚拟机也就退
出了。 因为没有了被守护者,Daemon也就没有工作可做了,也就没有继续运行程序的必要了。
重要:守护线程并非只有虚拟机内部提供,用户在编写程序时也可以自己设置守护线程。下面是用来设置守护线程的:
<font style=" line-height:26px;" face="'Microsoft YaHei'" size="3"><span style="font-size:12px;"><span style="font-family:'Microsoft YaHei';"><span style="font-family:SimSun;">Thread daemonTread = new Thread();
// 设定 daemonThread 为 守护线程,default false(非守护线程)
daemonThread.setDaemon(true);
// 验证当前线程是否为守护线程,返回 true 则为守护线程
daemonThread.isDaemon(); </span>
</span></span></font>
注意:
(1) thread.setDaemon(true)必须在thread.start()之前设置,否则会跑出一个IllegalThreadStateException异常。你不能把正在运行的常规线程设置为守护线程。
(2) 在Daemon线程中产生的新线程也是Daemon的。
(3) 不要认为所有的应用都可以分配给Daemon来进行服务,比如读写操作或者计算逻辑。
因为你不可能知道在所有的User完成之前,Daemon是否已经完成了预期的服务任务。一旦User退出了,可能大量数据还没有来得及读入或写出,计算任务也可能
多次运行结果不一样。这对程序是毁灭性的。造成这个结果理由已经说过了:一旦所有User Thread离开了,虚拟机也就退出运行了。
example:
//完成文件输出的守护线程任务
import java.io.*;
class TestRunnable implements Runnable{
public void run(){
try{
Thread.sleep(1000);//守护线程阻塞1秒后运行
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); //设置守护线程
thread.start(); //开始执行分进程
}
}
//运行结果:文件daemon.txt中没有"daemon"字符串。
查看txt文件,发现字符串并没有写入指定文件。原因很简单,直到主线程完成,守护线程仍处于1秒的阻塞状态。主线程很快运行结束,虚拟机退出,Daemon停止服务,输出操作自然失败了。
补充说明:
(1)定义:守护线程--也称“服务线程”,在没有用户线程可服务时会自动离开。
(2)优先级:守护线程的优先级比较低,用于为系统中的其它对象和线程提供服务。
(3)设置:通过setDaemon(true)来设置线程为“守护线程”;将一个用户线程设置为守护线程的方式是在 线程对象创建 之前 用线程对象的setDaemon方法。
(4)生命周期:守护进程(Daemon)是运行在后台的一种特殊进程。它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。也就是说守护线
程不依赖于终端,但是依赖于系统,与系统“同生共死”。Java的守护线程就是:当JVM中所有的线程都是守护线程的时候,JVM就可以退出了;如果还有一个
或以上的非守护线程则JVM不会退出。
(5)JRE判断程序是否执行结束的标准是所有的前台执线程行完毕,而不管后台线程的状态。
为什么需要守护进程:
我们知道静态变量是ClassLoader级别的,如果Web应用程序停止,这些静态变量也会从JVM中清除。但是线程则是JVM级别的,如果你在Web 应用中启动一个线
程,这个线程的生命周期并不会和Web应用程序保持同步。也就是说,即使你停止了Web应用,这个线程依旧是活跃的。正是因为这个很隐晦 的问题,所以很多有
经验的开发者不太赞成在Web应用中私自启动线程。
example:如果我们手工使用JDK Timer(Quartz的Scheduler),在Web容器启动时启动Timer,当Web容器关闭时,除非你手工关闭这个Timer,否则Timer中的任务还会继续运行!