• 守护线程与普通线程


    守护线程与普通线程的唯一区别是:当JVM中所有的线程都是守护线程的时候,JVM就可以退出了;如果还有一个或以上的非守护线程则不会退出。(以上是针对正常退出,调用System.exit则必定会退出)   

        所以setDeamon(true)的唯一意义就是告诉JVM不需要等待它退出,让JVM喜欢什么退出就退出吧,不用管它。

    守护线程在没有用户线程可服务时自动离开,在Java中比较特殊的线程是被称为守护(Daemon)线程的低级别线程。这个线程具有最低的优先级,用于为系统中的其它对象和线程提供服务。将一个用户线程设置为守护线程的方式是在线程对象创建之前调用线程对象的setDaemon方法。典型的守护线程例子是JVM中的系统资源自动回收线程,我们所熟悉的Java垃圾回收线程就是一个典型的守护线程,当我们的程序中不再有任何运行中的Thread,程序就不会再产生垃圾,垃圾回收器也就无事可做,所以当垃圾回收线程是Java虚拟机上仅剩的线程时,Java虚拟机会自动离开。它始终在低级别的状态中运行,用于实时监控和管理系统中的可回收资源。守护进程(Daemon)是运行在后台的一种特殊进程。它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。也就是说守护线程不依赖于终端,但是依赖于系统,与系统“同生共死”。那Java的守护线程是什么样子的呢。当JVM中所有的线程都是守护线程的时候,JVM就可以退出了;如果还有一个或以上的非守护线程则JVM不会退出。

    Java有两种Thread:“守护线程Daemon”与“用户线程User”。
    我们之前看到的例子都是用户,守护线程是一种“在后台提供通用性支持”的线程,它并不属于程序本体。
    从字面上我们很容易将守护线程理解成是由虚拟机(virtual machine)在内部创建的,而用户线程则是自己所创建的。事实并不是这样,任何线程都可以是“守护线程Daemon”或“用户线程User”。他们在几乎每个方面都是相同的,唯一的区别是判断虚拟机何时离开:
    用户线程:Java虚拟机在它所有非守护线程已经离开后自动离开。
    守护线程:守护线程则是用来服务用户线程的,如果没有其他用户线程在运行,那么就没有可服务对象,也就没有理由继续下去。
    setDaemon(boolean on)方法可以方便的设置线程的Daemon模式,true为Daemon模式,false为User模式。setDaemon(boolean on)方法必须在线程启动之前调用,当线程正在运行时调用会产生异常。isDaemon方法将测试该线程是否为守护线程。值得一提的是,当你在一个守护线程中产生了其他线程,那么这些新产生的线程不用设置Daemon属性,都将是守护线程,用户线程同样。
    复制代码
    package ying.thread;
    import java.io.IOException;
    
    public class Test extends Thread {
        public Test() { 
            
        } 
        
        public void run() { 
            for (int i = 0 ; i < 100 ; i ++) { 
                try {
                    Thread.sleep(100) ;
                } catch (InterruptedException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                } 
                System.out.println(i);   
            }
        } 
                
        public static void main (String args[]) { 
            Test test = new Test() ; 
            test.setDaemon(true) ; 
            test.start() ; 
            System.out.println("isDaemon=" + test.isDaemon()); 
            try {
                System.in.read() ;
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } 
        }
    }
    复制代码
    如果没有用户线程,那么守护线程也没有存活下去的意义了:
    复制代码
    package ying.thread;
    
    import java.io.IOException;
    
    public class Test extends Thread {
        public Test() { 
            
        } 
        
        public void run() { 
            for (int i = 0 ; i < 100 ; i ++) { 
                try {
                    Thread.sleep(100) ;
                } catch (InterruptedException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                } 
                System.out.println(i);   
            }
        } 
                
        public static void main (String args[]) { 
            Test test = new Test() ; 
            test.setDaemon(true) ; 
            test.start() ; 
            System.out.println("isDaemon=" + test.isDaemon()); 
        }
    }
    复制代码
    这个程序执行之后什么也不执行;什么也不打印;
    如果我们把 thread.setDaemon(true);删除,那么就可以打印出数字了。
    这就是守护线程,守护着最后一个用户线程,如果没有用户线程了,他也没作用了。不退出等什么??
  • 相关阅读:
    java:合并两个排序的链表(递归+非递归)
    BitSet: 有1千万个随机数,随机数的范围在1到1亿之间。现在要求写出一种算法,将1到1亿之间没有在随机数中的数求出来?
    【Java异常】Caused by: java.lang.IllegalArgumentException: method GET must not have a request body
    Java 语言规范要求 equals 方法具有的特性
    使用Java编写一个日历格式的方法
    【Java异常】The dependencies of some of the beans in the application context form a cycle
    Mysql字符串截取总结:left()函数、right()函数、substring()函数、substring_index()函数
    【Java异常】java.sql.SQLExcetion:Cannot convert value “0000-00-00 00:00:00” from column 9 to TIMESTAMP
    怎样将Beyond Compare添加到系统右键菜单
    【SVN异常】关于TortoiseSVNin目录下没有svn.exe执行程序文件的解决方案
  • 原文地址:https://www.cnblogs.com/baizhanshi/p/8289202.html
Copyright © 2020-2023  润新知