并发系列之「用户线程与守护线程」

守护线程 Daemon Thread

守护线程相对于正常线程来说,是比较特殊的一类线程,那么它特殊在哪里呢?别急,在了解它之前,我们需要知道一个问题,那就是:

JVM 程序在什么情况下能够正常退出?

The Java Virtual Machine exits when the only threads running are all daemon threads.

上面这句话来自 JDK 官方文档,意思是:

当 JVM 中不存在任何一个正在运行的非守护线程时,则 JVM 进程即会退出。

理解起来有点抽象,看下面的代码:

public static void main(String[] args) throws InterruptedException {
    
    
  	// 在主线程中 new 一个非守护线程
  	Thread thread = new Thread(()->{
    
      // ① 
      	// 模拟非守护线程不退出的情况
      	while(true){
    
                      // ②
          	try{
    
    
              	// 睡眠一秒
              	TimeUnit.SECONDS.sleep(1);
              	System.out.println("I am running ...");
            }catch(InterruptedException e){
    
    
              	e.printStackTrace();
            }
        }
    });
  
  	// 启动线程
  	thread.start();     // ③
  	TimeUnit.SECONDS.sleep(2);
  
  	// 主线程即将退出
  	System.out.println("the main thread ready to exit ...");  // ④
}
  • ①: 创建一个非守护线程;
  • ②: 模拟非守护线程不退出的情况;
  • ③: 启动线程;
  • ④: 主线程即将退出;

运行结果如下:

I am running ...
the main thread ready to exit ...
I am running ...
I am running ...
I am running ...

可以看到因为有一个非守护线程一直在后台运行着,JVM 无法正常退出。那么,如果说正在运行的是个守护线程,结果又会怎么样呢?

public static void main(String[] args) throws InterruptedException {
    
    
  	// 设置一个钩子线程,在 JVM 退出时输出日志
  	Runtime.getRuntime()
    .addShutdownHook(new Thread(()-> System.out.println("The JVM exit success !!!")) );
  
  	// 在主线程中 new 一个非守护线程
  	Thread thread = new Thread(()->{
    
      // ① 
      	// 模拟非守护线程不退出的情况
      	while(true){
    
                      // ②
          	try{
    
    
              	// 睡眠一秒
              	TimeUnit.SECONDS.sleep(1);
              	System.out.println("I am running ...");
            }catch(InterruptedException e){
    
    
              	e.printStackTrace();
            }
        }
    });
  
  	// 将线程设置为守护线程
    thread.setDaemon(true);
  
  	// 启动线程
  	thread.start();     // ③
  	TimeUnit.SECONDS.sleep(2);
  
  	// 主线程即将退出
  	System.out.println("the main thread ready to exit ...");  // ④
}
  • ①: 添加一个钩子(Hook)线程, 用来监听 JVM 退出,并输出日志;

  • ②: 通过 setDaemon(true)将该线程设置为守护线程;

运行结果:

I am running ...
the main thread ready to exit ...
The JVM exit success !!!

可以看到,当主线程退出时,JVM 会随之退出运行,守护线程同时也会被回收,即使你里面是个死循环也不碍事。

守护线程的作用及应用场景

通过上面的示例代码,相信你已经了解了守护线程和普通线程之间的区别,那么,我们来讨论一下为什么需要守护线程,以及何时使用,它的应用场景是什么?

上面,我们已经知道了,如果 JVM 中没有一个正在运行的非守护线程(全是守护线程),这个时候,JVM 会退出。换句话说,守护线程拥有自动结束自己生命周期的特性,而非守护线程不具备这个特点。

JVM 中的垃圾回收线程就是典型的守护线程,如果说不具备该特性,会发生什么呢?

当 JVM 要退出时,由于垃圾回收线程还在运行着,导致程序无法退出,这就很尴尬了!!!由此可见守护线程的重要性了。

通常来说,守护线程经常被用来执行一些后台任务,但是呢,你又希望在程序退出时,或者说 JVM 退出时,线程能够自动关闭,此时,守护线程是你的首选。

总结:如果你希望在主线程结束后JVM进程马上结束,那么在创建线程时可以将其设置为守护线程,如果你希望在主线程结束后子线程继续工作,等子线程结束后再让JVM进程结束,那么就将子线程设置为用户线程。

猜你喜欢

转载自blog.csdn.net/weixin_44471490/article/details/109019549
今日推荐