Java中的守护线程——daemon
絮叨
Java中有两类线程:User Thread(用户线程)、Daemon Thread(守护线程)
- 定义:守护线程(aka:服务线程),在没有用户线程可服务时会自动离开。
- 优先级:守护线程的优先级较低,用于为系统中的其它对象和线程提供服务。
用户线程即运行在前台的线程,而守护线程是运行在后台的线程。也就是说守护线程不依赖于终端,但是依赖于系统,与系统“同生共死”。守护线程作用是为其他前台线程的运行提供便利服务,而且仅在普通、非守护线程仍然运行时才需要,比如垃圾回收线程就是一个守护线程,当我们的程序中不再有任何运行的Thread,程序就不会再产生垃圾,垃圾回收器也就无事可做,所以当垃圾回收线程是JVM上仅剩的线程时,垃圾回收线程会自动离开。它始终在低级别的状态中运行,用于实时监控和管理系统中的可回收资源。当 VM 检测仅剩一个守护线程,而用户线程都已经退出运行时,VM就会退出,因为没有如果没有了被守护者,也就没有继续运行程序的必要了。如果有非守护线程仍然存活,VM 就不会退出。
守护线程并非只有虚拟机内部提供,用户在编写程序时也可以自己设置守护线程。用户可以用 Thread 的 setDaemon(true)方法设置当前线程为守护线程。
虽然守护线程可能非常有用,但必须小心确保其他所有非守护线程消亡时,不会由于它的终止而产生任何危害。因为你不可能知道在所有的用户线程退出运行前,守护线程是否已经完成了预期的服务任务。一旦所有的用户线程退出了,虚拟机也就退出运行了。 因此,不要在守护线程中执行业务逻辑操作(比如对数据的读写等)。
栗子🌰
package zhengbin.test; import java.util.Scanner; /** * Created by zhengbin06 on 2016/10/18. */ public class DaemonRunner implements Runnable{ @Override public void run() { while (true) { for (int i = 1;i <= 100;i++) { System.out.println(i); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } } } } public static void main(String[] args) { Thread daemonThread = new Thread(new DaemonRunner()); // 设置为守护线程 /** * 将该线程标记为守护线程或用户线程。当正在运行的线程都是守护线程时,Java虚拟机退出。 * 该方法必须在启动线程前调用。 * 该方法首先调用该线程checkAccess方法,且不带任何参数。这可能抛出SecurityException(在当前线程中)。 */ daemonThread.setDaemon(true); // 执行守护线程 daemonThread.start(); // isDaemon() 测试该线程是否为守护线程。 System.out.println("isDaemon = " + daemonThread.isDaemon()); Scanner scanner = new Scanner(System.in); // 接受输入,使程序在此停顿,一旦接受到用户输入,main线程结束,JVM退出! scanner.next(); //AddShutdownHook方法增加JVM停止时要做处理事件: //当JVM退出时,打印JVM Exit语句。 Runtime.getRuntime().addShutdownHook(new Thread() { /** * getRuntime() 返回与当前Java应用程序相关的运行时对象。Runtime类的大多数方法时实例方法,并且必须根据当前的运行时对象对其进行调用。 * addShutdownHook() 注册新的虚拟机来关闭钩子。 */ @Override public void run() { System.out.println("JVM Exit!"); } }); } }
上面的栗子中,先开启一个守护线程,每秒递增输出打印,同时main线程卡在了(阻塞)scanner,当控制台输入信息后,main线程结束,此时已经没有非守护线程的存在,守护线程也随之终止,停止输出,程序终止。
下面通过jps获得当前所有进程的pid,根据该pid,通过jstack查看该进程中的所有线程:
可以观察到,一个“main”,一个“Thread-0”。