什么是守护线程,它有什么特点
                           
天天向上
发布: 2025-07-12 13:44:54

原创
532 人浏览过

在 Java 中,守护线程(Daemon Thread) 是一种特殊的线程,主要用于为其他线程提供服务或后台支持,它的生命周期依赖于其他非守护线程(用户线程)


一、什么是守护线程?

守护线程是 JVM 提供的一类“后台线程”,当 Java 虚拟机中 所有的非守护线程(用户线程)都结束时,守护线程会自动退出,整个 JVM 进程终止。

守护线程的存在意义:不是为了完成任务,而是为了守护其他线程或系统资源。


二、守护线程与用户线程的区别

特性守护线程(Daemon)用户线程(User)
生命周期随所有用户线程结束而终止JVM 保证执行到任务完成
是否影响 JVM 退出否,所有用户线程结束后自动退出是,只要有用户线程在,JVM 继续运行
典型用途垃圾回收线程、定时调度、日志清理主业务逻辑执行线程

三、守护线程代码示例

public class DaemonExample {
    public static void main(String[] args) {
        Thread daemonThread = new Thread(() -> {
            while (true) {
                System.out.println("我是守护线程,后台运行中...");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException ignored) {}
            }
        });

        daemonThread.setDaemon(true);  // 设置为守护线程
        daemonThread.start();

        // 主线程休眠 3 秒后结束
        try {
            Thread.sleep(3000);
            System.out.println("主线程结束");
        } catch (InterruptedException ignored) {}
    }
}

输出(约 3 次):

我是守护线程,后台运行中...
我是守护线程,后台运行中...
我是守护线程,后台运行中...
主线程结束

主线程结束后,守护线程也立即停止,JVM 自动退出


四、设置守护线程的方法

必须 start() 方法调用之前 调用 setDaemon(true)

Thread t = new Thread(task);
t.setDaemon(true); // 设置为守护线程
t.start();

⚠️ 若在线程启动后再调用 setDaemon(true),将抛出 IllegalThreadStateException


五、守护线程的常见用途

应用场景示例说明
JVM 垃圾回收GC Thread 本身是守护线程
定时任务调度ScheduledExecutorService 中的线程
日志处理异步日志刷盘线程
后台资源清理如缓存过期清理、session清理线程
心跳监控线程定时发送心跳包维持连接状态

六、守护线程注意事项

  1. 不能依赖守护线程完成关键任务,如写文件、保存数据等
    → 因为 JVM 可能在它执行中途就退出了。
  2. 守护线程不能再启动用户线程
    守护线程只能“陪伴”用户线程,不能延长 JVM 生命周期。
  3. 线程池默认线程类型不是守护线程
    使用线程池时,如需后台执行,应手动设置为守护线程(见下文)。

七、线程池中设置守护线程方式

ExecutorService executor = Executors.newCachedThreadPool(new ThreadFactory() {
    public Thread newThread(Runnable r) {
        Thread t = new Thread(r);
        t.setDaemon(true); // 设置为守护线程
        return t;
    }
});

八、官方与权威资料


九、面试总结口诀

守护线程做后勤,主线程走它就停;别让它干正事,JVM 不保命。


更多详细内容请关注其他相关文章!

发表回复 0

Your email address will not be published. Required fields are marked *