守护线程 Daemon Thread 简介

介绍

在Java中有两类线程:User Thread(用户线程)、Daemon Thread(守护线程) 。
守护线程也称“服务线程”,守护线程会在没有用户线程可服务时自动离开。
守护线程的优先级比较低,用于为系统中的其它对象和线程提供服务。

Daemon的作用是为其他线程的运行提供便利服务,守护线程最典型的应用就是 GC (垃圾回收器),它就是一个很称职的守护者。

User和Daemon两者几乎没有区别,唯一的不同之处就在于虚拟机的离开:如果 User Thread已经全部退出运行了,只剩下Daemon Thread存在了,虚拟机也就退出了。 因为没有了被守护者,Daemon也就没有工作可做了,也就没有继续运行程序的必要了。

守护线程并非只有虚拟机内部提供,用户在编写程序时也可以自己设置守护线程。
Thread daemonTread = new Thread();  
daemonThread.setDaemon(true);   // 设定 daemonThread 为 守护线程
daemonThread.isDaemon();  // 验证当前线程是否为守护线程,返回 true 则为守护线程 
3
 
1
Thread daemonTread = new Thread();  
2
daemonThread.setDaemon(true);   // 设定 daemonThread 为 守护线程
3
daemonThread.isDaemon();  // 验证当前线程是否为守护线程,返回 true 则为守护线程 

守护线程使用注意

setDaemon(true) 必须在 start() 之前设置,否则会跑出一个 IllegalThreadStateException 异常,你不能把正在运行的常规线程设置为守护线程。

在Daemon线程中产生的新线程也是Daemon的。 
Thread thread = new Thread(() -> System.out.println(new Thread().isDaemon()));//true
thread.setDaemon(true);
thread.start();  
x
 
1
Thread thread = new Thread(() -> System.out.println(new Thread().isDaemon()));//true
2
thread.setDaemon(true);
3
thread.start();  

守护线程应该永远不去访问固有资源,如文件、数据库,因为它会在任何时候甚至在一个操作的中间发生中断。
public class Test {
	public static void main(String[] args) {
		Thread thread = new Thread(() -> {
			try {
				System.out.println("start");
				Thread.sleep(1000);//阻塞1秒后运行
				FileWriter writer = new FileWriter(new File("d://daemon.txt"), true);
				writer.write("daemon");
				writer.close();
				System.out.println("end");
			} catch (Exception e) {
				e.printStackTrace();
			}
		});
		thread.setDaemon(true); //设置守护线程时不会有任何内容写入文件
		thread.start(); //开始执行分进程
	}
}
x
1
public class Test {
2
    public static void main(String[] args) {
3
        Thread thread = new Thread(() -> {
4
            try {
5
                System.out.println("start");
6
                Thread.sleep(1000);//阻塞1秒后运行
7
                FileWriter writer = new FileWriter(new File("d://daemon.txt"), true);
8
                writer.write("daemon");
9
                writer.close();
10
                System.out.println("end");
11
            } catch (Exception e) {
12
                e.printStackTrace();
13
            }
14
        });
15
        thread.setDaemon(true); //设置守护线程时不会有任何内容写入文件
16
        thread.start(); //开始执行分进程
17
    }
18
}
结果,字符串并没有写入指定文件。原因很简单,直到主线程完成,守护线程仍处于1秒的阻塞状态,主线程很快就运行完了,然后虚拟机就退出了,Daemon停止服务,输出操作自然失败了。

守护线程的意义

用个比较通俗的比喻,任何一个守护线程都是整个JVM中所有非守护线程的保姆:只要当前JVM实例中尚存在任何一个非守护线程没有结束,守护线程就全部工作;只有当最后一个非守护线程结束时,守护线程随着JVM一同结束工作。

比如你正在用 Java 写成的编辑器写 Word 文档,你处理敲键盘事件的线程是个非守护线程, 除此之外,后台还有一个进行拼写检查的线程,它是个守护线程,他尽量不打扰你写稿子,他们可以同时进行,当守护线程发现有拼写错误时在状态条显示错误,但是你可以忽略。

比如城堡门前有个卫兵(守护线程),里面有诸侯(非守护线程),他们是可以同时干着各自的活儿,但是如果城堡里面的人都搬走了, 那么卫兵也就没有存在的意义了。

比如在使用长连接的comet服务端推送技术中,将消息推送线程设置为守护线程,服务于ChatServlet的servlet用户线程,在servlet的init启动消息线程,servlet一旦初始化后,一直存在服务器,servlet摧毁后,消息线程自动退出。

猜你喜欢

转载自www.cnblogs.com/baiqiantao/p/3416b9817af6287d35075209b556f459.html