Fork me on GitHub

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

posted @ 2016-10-18 11:59  郑斌blog  阅读(1448)  评论(0编辑  收藏  举报