共计 1541 个字符,预计需要花费 4 分钟才能阅读完成。
Java 中,通过 Thread 类,咱们能够创立 2 种线程,分为守护线程和用户线程。
守护线程是所有非守护线程的保姆,当所有非守护线程执行实现或退出了,即便还有守护线程在运行,JVM 也会间接退出,因而守护线程通常是用来解决一些辅助工作。
反之,对于非守护线程,只有有一个在运行,JVM 就不会退出。
典型的守护线程如垃圾回收 GC 线程,当用户线程都完结后,GC 也就没有独自存在的必要,JVM 间接退出。
咱们能够通过 Thread 对象的 setDaemon(boolean on)办法设置是否为守护线程,要在 start 之前设置:
Thread thread = new Thread(runnable);
thread.setDaemon(true); // true 示意守护线程,false 示意用户线程
thread.start();
须要留神的是,如果没有显示调用 setDaemon 办法进行设置,线程的模式是取决于父线程是否为守护线程 ,也就是创立此线程所在的线程。
如果父线程是守护线程,创立的线程默认是守护线程;
如果父线程是用户线程,创立的线程默认是用户线程。
这能够从 Thread 类的 init 办法源代码中看出:
Thread parent = currentThread();
this.daemon = parent.isDaemon();
对于 daemon 的设置,保留在了 Thread 对象的成员变量中,Thread 提供了 setter/getter:
private boolean daemon = false; // 是否为守护线程
public final void setDaemon(boolean on) {
// SecurityManager 安全检查,本文不展开讨论
checkAccess();
// 查看线程是否已启动,已启动无奈设置 daemon
if (isAlive()) {throw new IllegalThreadStateException();
}
daemon = on;
}
public final boolean isDaemon() {return daemon;}
setDaemon 办法中通过 isAlive 判断线程是否已启动,已启动状态下不容许批改,抛出 IllegalThreadStateException 异样。
接着咱们用示例来验证一下守护线程和非守护线程的区别。
以下是守护线程示例:
Thread t = new Thread(() -> {System.out.println("before");
ThreadUtil.sleep(5000);
System.out.println("after");
});
// 显式设置 daemon 为 true
t.setDaemon(true);
t.start();
ThreadUtil.sleep(1000);
System.out.println("exit");
输入:
before
exit
能够发现,当线程设置为守护线程后,主线程一旦执行结束,程序退出,守护线程也随着立刻终止。
以下是非守护线程示例:
Thread t = new Thread(() -> {System.out.println("before");
ThreadUtil.sleep(5000);
System.out.println("after");
});
// 显式设置 daemon 为 false
t.setDaemon(false);
t.start();
ThreadUtil.sleep(1000);
System.out.println("exit");
输入:
before
exit
after
尽管主线程曾经执行结束,但创立的非守护线程还在运行。
具体 JVM 是如何通过 daemon 字段控制线程的,这在 JDK 中找不到相应源码,须要深刻 hotspot C++ 源码进行剖析,后续有必要再追加更新。