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为truet.setDaemon(true);t.start();ThreadUtil.sleep(1000);System.out.println("exit");

输入:

beforeexit

能够发现,当线程设置为守护线程后,主线程一旦执行结束,程序退出,守护线程也随着立刻终止。

以下是非守护线程示例:

Thread t = new Thread(() -> {    System.out.println("before");    ThreadUtil.sleep(5000);    System.out.println("after");});//    显式设置daemon为falset.setDaemon(false);t.start();ThreadUtil.sleep(1000);System.out.println("exit");

输入:

beforeexitafter

尽管主线程曾经执行结束,但创立的非守护线程还在运行。

具体JVM是如何通过daemon字段控制线程的,这在JDK中找不到相应源码,须要深刻hotspot C++源码进行剖析,后续有必要再追加更新。