用户线程未执行完是否会阻止 JVM 终止?答案是:取决于线程类型。让我详细解释:

核心规则

JVM终止条件
所有用户线程结束
JVM终止
JVM继续运行
守护线程自动终止

线程类型与JVM终止关系

线程类型是否阻止JVM终止创建方式典型用途
用户线程✅ 是Thread t = new Thread();
t.setDaemon(false)
主线程、业务逻辑线程
守护线程❌ 否Thread t = new Thread();
t.setDaemon(true)
日志、监控、GC等后台任务

在您的日志系统中的体现

1. 日志处理器线程 (守护线程)

private PerfLogHandler() {Thread logProcessor = new Thread(this::processLogs, "perf-log-processor");logProcessor.setDaemon(true); // 设置为守护线程logProcessor.start();
}
  • 不会阻止JVM终止:当所有用户线程结束时自动终止
  • 可能丢失日志:JVM退出时未处理的队列日志会被丢弃

2. 线程池中的线程 (默认用户线程)

private final ThreadPoolExecutor logExecutor = new ThreadPoolExecutor(// ...new NamedThreadFactory("perf-log") // 默认创建用户线程
);

在您的 NamedThreadFactory 中:

public Thread newThread(Runnable r) {return new Thread(r, namePrefix + "-thread-" + threadNumber.getAndIncrement());// 默认是非守护线程(用户线程)
}

这里存在严重问题:这些线程会阻止JVM终止!

解决方案:优雅关闭

1. 修改线程工厂(创建守护线程)

private static class NamedThreadFactory implements ThreadFactory {@Overridepublic Thread newThread(Runnable r) {Thread t = new Thread(r, namePrefix + "-thread-" + threadNumber.getAndIncrement());t.setDaemon(true); // 关键修改!return t;}
}

2. 添加JVM关闭钩子(推荐)

private PerfLogHandler() {// 原初始化代码...// 添加关闭钩子Runtime.getRuntime().addShutdownHook(new Thread(() -> {shutdown();}));
}public void shutdown() {// 第一步:停止接收新日志logExecutor.shutdown();try {// 第二步:等待现有任务完成(最多10秒)if (!logExecutor.awaitTermination(10, TimeUnit.SECONDS)) {// 第三步:强制取消未完成任务List<Runnable> skipped = logExecutor.shutdownNow();log.warn("Forced shutdown, skipped {} logs", skipped.size());}// 第四步:处理队列剩余日志List<String> remaining = new ArrayList<>();logQueue.drainTo(remaining);if (!remaining.isEmpty()) {log.info("Processing {} remaining logs", remaining.size());new LogBatchTask(remaining).run();}} catch (InterruptedException e) {Thread.currentThread().interrupt();}
}

完整生命周期示例

用户线程JVM日志守护线程启动应用启动守护线程添加日志到队列执行业务逻辑所有用户线程结束发送中断信号处理剩余日志线程终止退出用户线程JVM日志守护线程

关键注意事项

  1. 守护线程使用原则

    • 只用于非关键后台任务
    • 必须能随时安全终止
    • 不能持有需要关闭的资源(如数据库连接)
  2. 用户线程阻塞场景

    public static void main(String[] args) {new Thread(() -> {while(true) { // 无限循环将阻止JVM退出}}).start();
    }
    
  3. 最佳实践

    • 后台服务线程设为守护线程
    • 使用ExecutorService替代裸线程
    • 总是实现优雅关闭逻辑
    • 监控线程状态:
      ThreadMXBean threadBean = ManagementFactory.getThreadMXBean();
      long[] ids = threadBean.getAllThreadIds();
      for (long id : ids) {ThreadInfo info = threadBean.getThreadInfo(id);System.out.println(info.getThreadName() + " is daemon: " + info.isDaemon());
      }
      

结论

在您的日志系统中:

  • 日志处理线程(守护线程)不会阻止JVM终止
  • 线程池工作线程(默认用户线程)会阻止JVM终止
  • 必须将线程池线程设为守护线程或实现优雅关闭

正确配置后,当所有业务线程(用户线程)结束时:

  1. JVM会终止所有守护线程
  2. 日志系统自动关闭
  3. JVM正常退出

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/92401.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/92401.shtml
英文地址,请注明出处:http://en.pswp.cn/web/92401.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux Vim 常用快捷键

Vim中最常用的快捷键&#xff0c;熟练掌握它们可以大大提高编辑效率。移动光标h- 左移j- 下移k- 上移l- 右移w- 移动到下一个单词开头b- 移动到上一个单词开头e- 移动到单词末尾0- 移动到行首$- 移动到行尾gg- 移动到文件开头G- 移动到文件末尾:n- 跳转到第n行插入模式i- 在光标…

【Bellman负环】Cycle Finding

题目翻译给定一个有向图&#xff0c;你的任务是判断它是否包含负环&#xff0c;并给出这样一个环的示例。输入 第一行输入两个整数 n 和 m&#xff1a;分别表示节点数和边数。节点编号为 1, 2, ..., n。 接下来 m 行描述边&#xff0c;每行有三个整数 a, b, c&#xff1a;表示存…

数据结构(六):树与二叉树

一、树的基本概念树的定义树&#xff08;Tree&#xff09;是由 n&#xff08;n ≥ 0&#xff09;个节点组成的有限集合&#xff0c;当 n 0 时称为空树。非空树中&#xff1a;有且仅有一个根节点&#xff08;Root&#xff09;&#xff1b;其余节点可以划分为若干个互不相交的子…

《Linux运维总结:Shell 脚本日志输出工具》

总结&#xff1a;整理不易&#xff0c;如果对你有帮助&#xff0c;可否点赞关注一下&#xff1f; 更多详细内容请参考&#xff1a;Linux运维实战总结 一、Shell 脚本日志输出工具 1、提供的 logger() 函数是一个非常实用的 Shell 脚本日志输出工具&#xff0c;它支持带时间戳和…

select ... for update阻塞

总结阻塞规则&#xff1a;当前事务持有的锁 (来自 SELECT ... FOR UPDATE)其他事务尝试的操作是否会被阻塞&#xff1f;原因排他锁 (X Lock) 在行 R 上SELECT ... FROM ... (普通查询)否读快照 (MVCC)&#xff0c;不需要锁排他锁 (X Lock) 在行 R 上SELECT ... FROM ... FOR UP…

LangChain4j终极指南:Spring Boot构建企业级Agent框架

LangChain4j Spring Boot 构建企业级 Agent 框架深度指南&#xff08;3000字终极版&#xff09;一、架构设计&#xff1a;面向未来的企业级智能体系统1.1 分层架构设计1.2 核心组件职责1.3 企业级特性设计二、核心模块深度实现2.1 智能体协作引擎&#xff08;LangGraph4j高级应…

前端基础之《Vue(29)—Vue3 路由V4》

一、安装1、命令cnpm install vue-router42、配置映射为src路径&#xff08;1&#xff09;安装对应配置cnpm install types/node&#xff08;2&#xff09;配置vite.config.tsimport { defineConfig } from vite import vue from vitejs/plugin-vue import * as path from &quo…

9.2 通过DuEDrawingControl把eDrawing嵌入到C#中显示

本文介绍如何通过DuEDrawingControl控件在C#的WPF中进行3D的显示。 DuEDrawingControl在实际应用中可以应用于以下场景: 1.CAD文件预览:在Winform或WPF应用程序中,用户可以预览装配文件、工程图文件等,方便进行设计和审核。 2.打印管理:控件支持打印文件的管理,用…

《Vuejs设计与实现》第 13 章(异步组件和函数式组件

目录 13.1 异步组件的问题与解决方法 13.2 异步组件的实现原理 3.2.1 封装 defineAsyncComponent 函数 13.2.2 超时与 Error 组件 13.2.3 延迟与 Loading 组件 13.2.4 重试机制 13.3 函数式组件 13.4 总结 在第12章&#xff0c;我们深入探讨了组件的基本含义和实现方式…

Python的七大框架对比分析

谈到“Python 七大框架”时&#xff0c;通常指 Django、Flask、FastAPI、Tornado、Sanic、AIOHTTP 和 Pyramid 这七位“常驻嘉宾”。它们各有气质&#xff0c;适合的场景也截然不同。1. DjangoDjango 像一辆全副武装的重型越野&#xff1a;出厂就配好 ORM、后台管理、权限、缓存…

Redis中String数据结构为什么以长度44为embstr和raw实现的分界线?

​ 一道常见Redis面试题。 ​ 在Redis的String数据结构中&#xff0c;当字符串的实际长度小于44且包含非整数字符时底层编码方式为embstr。当超过44时使用raw底层编码方式。 ​ 那么为什么要以字符串的长度44为分界线呢&#xff1f; 信息一 ​ 首先要分析embst…

告别人工巡查,校园空调管理迈入智能物联高效时代

在“双碳”战略深入推进和智慧校园建设加速落地的背景下&#xff0c;学校空调的用电管理已经不再是“开与关”的简单问题&#xff0c;而是涵盖了能效优化、安全保障、智慧化管理的综合课题。蓝奥声科技凭借LPIOT低功耗物联网、ECWAN边缘协同网络等优势技术&#xff0c;打造出面…

Access开发右下角浮窗提醒

Hi&#xff0c;大家好呀&#xff01;感觉又有很长一段时间没有给大家更新内容了&#xff0c;最近一直在忙&#xff0c;给大家承诺的框架、视频教程、直播等等感觉又要跳票了&#xff0c;嘿嘿&#xff0c;但大家还是不要急&#xff0c;莫催我&#xff0c;我会慢慢都更新出来的&a…

AI自进化,GPU性能翻三倍——CUDA-L1开启自动优化新范式

最近看到一篇让我挺震撼的文章&#xff0c;来自 DeepReinforce 团队发布的一个新框架——CUDA-L1。说实话&#xff0c;刚看到标题说“AI 让 GPU 性能提升 3 倍以上”&#xff0c;我心里是有点怀疑的。毕竟我们搞科研的都知道&#xff0c;这种宣传语很多时候水分不小。但当我静下…

深入理解 Java AWT Container:原理、实战与性能优化

以 Container 为核心梳理 AWT 容器体系与事件模型&#xff0c;提供可运行的纯 AWT 示例&#xff08;含 Panel、Frame、Dialog、ScrollPane 正确用法&#xff09;&#xff0c;并给出常见问题与性能优化建议。 Java AWT, Container, 容器, 布局管理器, 事件驱动, ScrollPane, 性…

redis--黑马点评--用户签到模块详解

用户签到假如我们使用一张表来存储用户签到信息&#xff0c;其结构应该如下&#xff1a;CREATE TABLE tb_sign (id bigint unsigned NOT NULL AUTO_INCREMENT COMMENT 主键,user_id bigint unsigned NOT NULL COMMENT 用户id,year year NOT NULL COMMENT 签到的年,month tinyin…

Shell、Python对比

在 Shell 脚本&#xff08;sh/bash&#xff09; 和 Python 之间选择时&#xff0c;主要取决于具体的使用场景和需求。以下是两者的对比分析&#xff0c;帮助你判断哪种更方便&#xff1a;1. Shell 脚本&#xff08;sh/bash&#xff09;的优势适用场景系统管理任务&#xff1a;如…

自适应反步控制:理论与设计

自适应反步控制 文章目录自适应反步控制1. 基本思想A. 第一步B. 第二步1. 基本思想 基于传统反步法&#xff0c;考虑了系统方程中以线性形式出现的未知参数。核心思想包括参数估计率和控制率。 考虑二阶系统&#xff1a; {x˙1x2φ1T(x1)θx˙2uφ2T(x1,x2)θ(1)\begin{cases…

[Oracle] LEAST()函数

LEAST() 是 Oracle 中一个非常有用的函数&#xff0c;用于从一组表达式中返回最小值LEAST()函数会从给定的参数列表中返回最小的值&#xff0c;它与GREATEST()函数正好相反语法格式LEAST(expr1, expr2 [, expr3, ...])参数说明expr1, expr2, ...&#xff1a;要比较的表达式(至少…

SVM算法实战应用

目录 用 SVM 实现鸢尾花数据集分类&#xff1a;从代码到可视化全解析 一、算法原理简述 二、完整代码实现 三、代码解析 1. 导入所需库 2. 加载并处理数据 3. 划分训练集和测试集 4. 训练 SVM 模型 5. 计算决策边界参数 6. 生成决策边界数据 7. 绘制样本点 8. 绘制…