- Java的线程分用户线程(非守护线程)和守护线程。
- java虚拟机要等用户线程都结束以后才启动关闭过程。
- 而守护线程的角色是为了用户线程提供服务。因此只有当用户线程存在,守护线程才有存在的必要。守护线程不会阻止Java虚拟机启动关闭过程。
- 当用户线程都结束以后,守护线程也不再执行,所以在守护线程中使用无限循环不会引起问题。不推荐在守护线程中执行I/O任务。
- 可以使用Thread的setDaemon函数设置是否守护线程。注意:这个方法必须在线程已经创建、但没有启动之前调用。
下面的代码示例,应用的主线程是用户线程,另外创建了一个守护线程:
package com.thb;
import java.util.concurrent.TimeUnit;
public class Test6 {
public static void main(String[] args) {
// 创建一个线程
Thread daemonThread = new Thread(() -> {
// 打印该线程的优先级
System.out.println("[" + Thread.currentThread().getName() + "]" + " priority is " + Thread.currentThread().getPriority());
while (true) {
System.out.println("in thread: " + Thread.currentThread().getName());
try {
// 睡眠2秒钟
TimeUnit.SECONDS.sleep(2);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}, "daemon thread");
// 设置线程是守护线程
daemonThread.setDaemon(true);
// 启动守护线程
daemonThread.start();
// 打印主线程的优先级
System.out.println("[" + Thread.currentThread().getName() + "]" + " priority is " + Thread.currentThread().getPriority());
for (int i = 0; i < 3; i++) {
try {
System.out.println("in thread: " + Thread.currentThread().getName());
// 睡眠3秒钟
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
一次运行输出:
[daemon thread] priority is 5
[main] priority is 5
in thread: main
in thread: daemon thread
in thread: daemon thread
in thread: main
in thread: daemon thread
in thread: main
in thread: daemon thread
in thread: daemon thread
从输出可以看出,守护线程中尽管使用了无限循环,但它并没有一直运行下去。