首页 > 代码库 > Java中的守护线程

Java中的守护线程

絮叨

  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”。

Java中的守护线程