说说用户线程和守护线程
用户线程和守护线程了解吗?
什么是用户线程和守护线程?
守护线程是一种特殊的线程,在后台默默地完成一些系统性的服务,比如垃圾回收线程、JIT线程都是守护线程。与之对应的是用户线程,用户线程可以理解为是系统的工作线程,它会完成这个程序需要完成的业务操作。
如何手动设置线程为守护线程?
java 中的线程分为两种:守护线程(Daemon)和用户线程(User)。任何线程都可以设置为守护线程和用户线程,通过方法 setDaemon() 即可实现。
// 接口方法
void setDaemon(boolean on) // 将此线程标记为用户线程,true 则把该线程设置为守护线程,反之则为用户线程
boolean isDaemon() // 判断这个线程是否是守护线程,返回true表示守护线程,否则为用户线程
注意点:
- 当程序中所有的用户线程执行完毕之后,不管守护线程是否结束,系统都会自动退出;
- java线程分为用户线程和守护线程,线程的 daemon 属性为 true 表示是守护线程,false 表示是用户线程;
- 设置守护线程:
t1.setDaemon(true)
,该语句要放在t1.start()
方法执行之前,如果放在后面,会报IllegalThreadStateException
异常,不起作用;
🌰 1:
public class Demo {
/**
验证:当程序中所有的用户线程执行完毕之后,不管守护线程是否结束,系统都会自动退出
*/
static class T1 extends Thread {
public T1(String name) {
super(name);
}
@Override
public void run() {
System.out.println(this.getName() + "开始执行," + (this.isDaemon() ? "我是守护线程" : "我是用户线程"));
while (true) ; // 死循环
}
}
public static void main(String[] args) {
T1 t1 = new T1("t1");
t1.setDaemon(true); // 放在 t1.start() 方法执行之前,设为守护线程
t1.start();
System.out.println("主线程结束");
}
}
//------------------ 运行结果
主线程结束
t1开始执行,我是守护线程 (程序正常结束)
🌰 2:
/**
验证:设置守护线程,需要在start()方法之前进行
*/
public class Demo1 {
static void main(String[] args) {
Thread t1 = new Thread() {
@Override
public void run() {
try {
TimeUnit.SECONDS.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
t1.start();
t1.setDaemon(true);
}
}
//------------------ 运行结果
Exception in thread "main" java.lang.IllegalThreadStateException
at java.lang.Thread.setDaemon(Thread.java:1359)
源码看看
查看 Thread 的源码,发现 布尔型变量 daemon 是Thread类中的一个成员变量,默认值为 false。
/* Thread.java */
public class Thread implements Runnable {
......
// 布尔型变量 daemon 是Thread类中的一个成员变量,默认值为 false
private boolean daemon = false;
}
查看线程的 init
方法(init 方法在创建线程时会执行,该方法作用在Thread
构造器内部),可以看出 dameon 的默认值为父线程的daemon。也就是说,父线程如果为用户线程,子线程默认也是用户现场;父线程如果是守护线程,子线程默认也是守护线程。
父线程就是创建当前线程的那个线程,比如我们经常在main函数中 new 一个 t1 线程,那 main 线程是 t1 线程的父线程,t1 是子线程。
/* Thread.java */
private void init(ThreadGroup g, Runnable target, String name,.......){
......
Thread parent = currentThread();
this.daemon = parent.isDaemon();
......
}
🌰 1:
public class Demo2 {
static class T extends Thread {
public T(String name) {
super(name);
}
@Override
public void run() {
System.out.println(this.getName() + ".daemon:" + this.isDaemon());
}
}
public static void main(String[] args) throws InterruptedException {
// 打印 main 线程的状态
System.out.println(Thread.currentThread().getName() + ".daemon:" + Thread.currentThread().isDaemon());
// main 线程中创建了 t1 线程
T t1 = new T("t1");
t1.start();
Thread t2 = new Thread() {
@Override
public void run() {
System.out.println(this.getName() + ".daemon:" + this.isDaemon());
// t2 内创建了 t3 线程
T t3 = new T("t3");
t3.start();
}
};
t2.setName("t2");
t2.setDaemon(true); // 手动设置 t2 为守护线程
t2.start();
TimeUnit.SECONDS.sleep(2);
}
}
//------------------ 运行结果
main.daemon:false
t1.daemon:false
t2.daemon:true
t3.daemon:true
总结
- java 中的线程分为两种:守护线程(Daemon)和用户线程(User);
- 当程序中所有的用户线程执行完毕之后,不管守护线程是否结束,系统都会自动退出;
- 任何线程都可以设置为守护线程和用户线程,通过方法
setDaemon(boolean on)
即可实现。此外,setDaemon(boolean on)
方法必须在线程的start()方法之前调用,在后面调用会报异常; - 线程的
daemon
默认值和其父线程(创建它的线程)一样; - Thread Dump 打印出来的线程信息,含有 daemon 字样的线程即为守护进程,常见的守护线程有 服务守护进程、编译守护进程、windows 下的监听 Ctrl+break 的守护进程、Finalizer 守护进程、引用处理守护进程、GC 守护进程。