孤舟笔记 并发篇十六 什么是守护线程?它有什么特点?面试官问这个是在考察你的JVM底层认知
文章目录一、先说结论守护线程核心特点二、用户线程 vs 守护线程三、守护线程最危险的坑finally 不靠谱四、守护线程用在哪五、setDaemon 的两个注意点守护线程全景回答技巧与点评标准回答加分回答面试官点评个人网站你有没有遇到过这种情况程序明明跑完了但 JVM 就是不退出非得手动 kill或者反过来后台日志线程还在写JVM 却直接退出了日志丢了这些行为都和守护线程有关。面试官问守护线程不是让你背定义而是考你JVM 什么时候退出这个底层认知。一、先说结论守护线程核心特点维度说明定义为其他线程提供服务的后台线程设置方式thread.setDaemon(true)必须在 start() 前设置生命周期所有用户线程结束时守护线程自动销毁JVM 退出条件所有用户线程执行完毕不管守护线程是否还在运行典型用途GC 线程、JIT 编译线程、监控线程、日志线程注意事项守护线程中的 finally 块不一定会执行一句话记住守护线程就像商场的保安——顾客用户线程都走了保安自动下班不会一个人在空商场里待着。二、用户线程 vs 守护线程Java 线程分两种用户线程和守护线程。区别就一个JVM 等不等它。// 用户线程默认Threadt1newThread(()-{while(true){System.out.println(用户线程运行中...);}});t1.start();// JVM 永远不会退出因为 t1 是用户线程 // 守护线程Threadt2newThread(()-{while(true){System.out.println(守护线程运行中...);}});t2.setDaemon(true);// 必须在 start() 前设置 t2.start();// 主线程结束后 JVM 直接退出t2 被强制终止 生活类比用户线程是餐厅里的客人守护线程是服务员。客人全走了服务员直接下班不会继续在空餐厅里端盘子。三、守护线程最危险的坑finally 不靠谱守护线程被 JVM 强制终止时finally 块可能不会执行ThreaddaemonnewThread(()-{try{while(true){// 模拟日志写入}}finally{// ⚠️ 这段代码不一定会执行logFile.close();// 可能来不及关闭 }});daemon.setDaemon(true);daemon.start();当所有用户线程结束JVM 退出时直接杀掉守护线程不会等它执行完 finally。这就好比商场突然停电保安来不及锁门就被赶走了。正确做法需要确保资源释放的操作不要放在守护线程里或者使用 shutdown hookRuntime.getRuntime().addShutdownHook(newThread(()-{logFile.close();// JVM 退出前保证执行 }));四、守护线程用在哪JDK 自己就在大量使用守护线程场景线程说明垃圾回收GC 线程为用户线程回收内存自然应该是守护线程即时编译JIT 编译线程优化热点代码不影响 JVM 退出监控JMX 线程监控 JVM 状态用户线程没了监控也没意义日志Log4j AsyncAppender后台异步写日志共同点它们都是辅助角色没有它们程序照样能跑只是变慢或没日志它们的存在是为了服务用户线程。五、setDaemon 的两个注意点// ❌ 在 start() 之后设置会抛异常ThreadtnewThread(task);t.start();t.setDaemon(true);// IllegalThreadStateException // ❌ 守护线程创建的子线程默认也是守护线程ThreaddaemonnewThread(()-{ThreadchildnewThread(()-{// 这个子线程也是守护线程 });child.start();});daemon.setDaemon(true);关键规则守护线程的守护属性会遗传给它的子线程。这也合理——如果父线程是服务员子线程也应该是服务员否则客人走了子线程还在那跑JVM 就退不出了。守护线程全景守护线程 全景 核心概念 ├── 用户线程JVM 必须等它结束才退出 └── 守护线程JVM 不等用户线程走完就销毁 关键特性 ├── setDaemon(true) 必须在 start() 前调用 ├── 所有用户线程结束 → 守护线程强制终止 ├── finally 块不保证执行 └── 子线程继承守护属性 典型应用 ├── GC 线程、JIT 编译线程 ├── JMX 监控线程 └── 日志异步写入线程 避坑指南 ├── 不要在守护线程中做资源释放 ├── 用 ShutdownHook 保证清理 └── 不要依赖守护线程的执行完整性 口诀守护线程为他人用户走完它就停 finally 不可靠start 前设才生效 子线程随父属性资源释放用 Hook。回答技巧与点评标准回答守护线程是为用户线程提供后台服务的线程。通过 setDaemon(true) 设置必须在 start() 前调用。当所有用户线程执行完毕后JVM 会退出不管守护线程是否还在运行。典型应用包括 GC 线程、JIT 编译线程等。需要注意守护线程的 finally 块不一定会执行因此不适合做资源释放操作。加分回答JVM 退出机制JVM 退出的判断条件是所有非守护线程都结束了而不是所有线程都结束。这是守护线程存在的根本意义——JVM 不会为一个后台服务线程而赖着不走属性继承守护线程创建的子线程默认也是守护线程这个遗传机制保证了不会因为子线程是用户线程而导致 JVM 无法退出实际陷阱数据库连接池的心跳线程如果设为守护线程主线程结束时连接可能来不及归还就强杀了日志异步线程如果是守护线程最后几条日志可能丢失。所以生产环境中选择守护线程要谨慎需要评估丢失操作的后果面试官点评这道题考的是你对 JVM 生命周期管理的理解。能说出定义只是入门能讲清 finally 不可靠、子线程继承、以及实际应用中的陷阱才算真正理解。面试官最想听到的答案是你不仅知道守护线程是什么还知道什么时候不该用守护线程。原文阅读内容有帮助点赞、收藏、关注三连评论区等你
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2570981.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!