• 说说用户线程和守护线程


    用户线程和守护线程了解吗?

    什么是用户线程和守护线程?

    守护线程是一种特殊的线程,在后台默默地完成一些系统性的服务,比如垃圾回收线程JIT线程都是守护线程。与之对应的是用户线程,用户线程可以理解为是系统的工作线程,它会完成这个程序需要完成的业务操作。

    如何手动设置线程为守护线程?

    java 中的线程分为两种:守护线程(Daemon)和用户线程(User)。任何线程都可以设置为守护线程和用户线程,通过方法 setDaemon() 即可实现。

    // 接口方法
    void setDaemon(boolean on)	// 将此线程标记为用户线程,true 则把该线程设置为守护线程,反之则为用户线程
        
    boolean	isDaemon()		// 判断这个线程是否是守护线程,返回true表示守护线程,否则为用户线程
    

    注意点:

    • 当程序中所有的用户线程执行完毕之后,不管守护线程是否结束,系统都会自动退出;
    • java线程分为用户线程和守护线程,线程的 daemon 属性为 true 表示是守护线程,false 表示是用户线程;
    • 设置守护线程:t1.setDaemon(true),该语句要放在 t1.start() 方法执行之前,如果放在后面,会报 IllegalThreadStateException 异常,不起作用;

    🌰 1:

    public class Demo {
        /**
        	验证:当程序中所有的用户线程执行完毕之后,不管守护线程是否结束,系统都会自动退出
        */
        
        static class T1 extends Thread {
            public T1(String name) {
                super(name);
            }
            @Override
            public void run() {
                System.out.println(this.getName() + "开始执行," + (this.isDaemon() ? "我是守护线程" : "我是用户线程"));
                while (true) ;	// 死循环
            }
        }
        public static void main(String[] args) {
            T1 t1 = new T1("t1");
            t1.setDaemon(true);		// 放在 t1.start() 方法执行之前,设为守护线程
            t1.start();
            System.out.println("主线程结束");
        }
    }
    
    //------------------ 运行结果
    主线程结束
    t1开始执行,我是守护线程		(程序正常结束)

    🌰 2:

    /**
    	验证:设置守护线程,需要在start()方法之前进行
    */
    public class Demo1 {
        
        static void main(String[] args) {
            Thread t1 = new Thread() {
                @Override
                public void run() {
                    try {
                        TimeUnit.SECONDS.sleep(10);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            };
            t1.start();
            t1.setDaemon(true);
        }
    }
    
    //------------------ 运行结果
    Exception in thread "main" java.lang.IllegalThreadStateException
    	at java.lang.Thread.setDaemon(Thread.java:1359)

    源码看看

    查看 Thread 的源码,发现 布尔型变量 daemon 是Thread类中的一个成员变量,默认值为 false。

    /*  Thread.java  */
    public class Thread implements Runnable {
     	......
    	// 布尔型变量 daemon 是Thread类中的一个成员变量,默认值为 false
        private boolean daemon = false;
    }

    查看线程的 init 方法(init 方法在创建线程时会执行,该方法作用在Thread构造器内部),可以看出 dameon 的默认值为父线程的daemon。也就是说,父线程如果为用户线程,子线程默认也是用户现场;父线程如果是守护线程,子线程默认也是守护线程。

    父线程就是创建当前线程的那个线程,比如我们经常在main函数中 new 一个 t1 线程,那 main 线程是 t1 线程的父线程,t1 是子线程。

    /*  Thread.java  */
    private void init(ThreadGroup g, Runnable target, String name,.......){
        ......
        Thread parent = currentThread();
    	this.daemon = parent.isDaemon();
        ......
    }

    🌰 1:

    public class Demo2 {
        
        static class T extends Thread {
            public T(String name) {
                super(name);
            }
            @Override
            public void run() {
                System.out.println(this.getName() + ".daemon:" + this.isDaemon());
            }
        }
        
        public static void main(String[] args) throws InterruptedException {
            // 打印 main 线程的状态
            System.out.println(Thread.currentThread().getName() + ".daemon:" + Thread.currentThread().isDaemon());
            // main 线程中创建了 t1 线程
            T t1 = new T("t1");
            t1.start();
            
            Thread t2 = new Thread() {
                @Override
                public void run() {
                    System.out.println(this.getName() + ".daemon:" + this.isDaemon());
                    // t2 内创建了 t3 线程
                    T t3 = new T("t3");
                    t3.start();
                }
            };
            t2.setName("t2");
            t2.setDaemon(true);		// 手动设置 t2 为守护线程
            t2.start();
            TimeUnit.SECONDS.sleep(2);
        }
    }
    
    //------------------ 运行结果
    main.daemon:false	
    t1.daemon:false
    t2.daemon:true
    t3.daemon:true

    总结

    1. java 中的线程分为两种:守护线程(Daemon)和用户线程(User);
    2. 当程序中所有的用户线程执行完毕之后,不管守护线程是否结束,系统都会自动退出;
    3. 任何线程都可以设置为守护线程和用户线程,通过方法 setDaemon(boolean on) 即可实现。此外,setDaemon(boolean on)方法必须在线程的start()方法之前调用,在后面调用会报异常;
    4. 线程的daemon默认值和其父线程(创建它的线程)一样;
    5. Thread Dump 打印出来的线程信息,含有 daemon 字样的线程即为守护进程,常见的守护线程有 服务守护进程、编译守护进程、windows 下的监听 Ctrl+break 的守护进程、Finalizer 守护进程、引用处理守护进程、GC 守护进程。

    参考

    http://itsoku.com/course/1/9#目录


    __EOF__

  • 本文作者: #阿飞的客栈#
  • 本文链接: https://www.cnblogs.com/afei688/p/16618018.html
  • 关于博主: 评论和私信会在第一时间回复。或者直接私信我。
  • 版权声明: 本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
  • 声援博主: 如果您觉得文章对您有帮助,可以点击文章右下角推荐一下。
  • 相关阅读:
    P5661 [CSP-J2019] 公交换乘
    树莓派4B搭建homeassistant 物联网平台
    SpringMVC总结
    实战:如何优雅的从 Skywalking 切换到 OpenTelemetry
    【JAVA后端开发】Part1--瑞吉外卖项目
    PolarDB-X 源码解读系列:DML 之 INSERT IGNORE 流程
    Acwing 286. 选课(背包类树形DP)
    封装Ajax--解决地狱回调问题
    Scala基础【模板方法设计模式、基础面向对象编程】
    【探索Linux】—— 强大的命令行工具 P.14(进程间通信 | 匿名管道 | |进程池 | pipe() 函数 | mkfifo() 函数)
  • 原文地址:https://www.cnblogs.com/afei688/p/16618018.html