springboot项目不退出的原因

一、普通java程序vs springboot项目的本质

维度 普通 Java 程序(如 java HelloWorld Spring Boot 服务
运行模式 执行完 main() 就结束 main() 启动后阻塞等待,不结束
生命周期 秒级/分钟级 天级/月级(7×24 运行)
与 JVM 关系 JVM 随程序结束而退出 JVM 长期存活,请求在已运行的 JVM 内处理
类加载时机 启动时一次性加载 启动时加载核心类,运行时可能懒加载
JIT 编译时机 运行期间触发 运行期间持续触发,服务越久性能越好

二、Spring Boot 为什么能"一直运行不结束"?

核心:非守护线程阻塞

java 复制代码
// SpringApplication.run() 的核心逻辑(简化)
public ConfigurableApplicationContext run(String... args) {
    // 1. 创建 Spring 上下文
    ConfigurableApplicationContext context = createApplicationContext();
    
    // 2. 刷新上下文(加载 Bean、启动内嵌 Tomcat)
    refreshContext(context);
    
    // 3. 发布启动完成事件
    listeners.started(context);
    
    // 4. 调用 Runner(CommandLineRunner/ApplicationRunner)
    callRunners(context, args);
    
    // 5. 返回上下文,但 main 线程在这里做了什么?
    return context;
}

// 这里需要再去学习下springboot的启动过程

关键点:内嵌 Tomcat/Jetty/Undertow 启动了非守护线程。

java 复制代码
// Tomcat 启动时会创建线程池处理请求
// 这些线程默认是 非守护线程(non-daemon)
Executor executor = new ThreadPoolExecutor(
    corePoolSize, 
    maxPoolSize, 
    keepAliveTime, 
    TimeUnit.SECONDS, 
    new LinkedBlockingQueue<>(),
    new CustomThreadFactory()  // 默认创建的是非守护线程
);

JVM 退出条件:所有非守护线程结束。

java 复制代码
main 线程启动
    │
    ▼
启动 Tomcat(创建非守护线程:Acceptor、Poller、Worker 线程)
    │
    ▼
main 线程执行完 run() 方法
    │
    ▼
main 线程结束,但 JVM 不退出
    │
    ▼
Tomcat 的 Acceptor 线程仍在监听端口(如 8080)
    │
    ▼
有新请求 → Worker 线程从线程池取出处理
    │
    ▼
请求处理完 → Worker 线程归还线程池,**不销毁 JVM**

只要还有非守护线程在运行,JVM 就不会退出。Spring Boot 通过内嵌 Web 服务器维持这些线程。

相关推荐
devilnumber1 小时前
Java 二分查找(二分算法)详解 + 实战运用 + 核心坑点
java·开发语言·算法
仍然.1 小时前
SpringBoot快速上手
java·spring boot·后端
ch.ju1 小时前
Java程序设计(第3版)第四章——重载和覆盖的区别
java·开发语言
浮尘笔记2 小时前
Go实现大文件异步流式采集引擎
开发语言·后端·golang
霸道流氓气质2 小时前
Spring Boot 大数据量 Excel 导入导出功能实现指南
spring boot·后端·excel
仍然.2 小时前
Spring MVC(1)---介绍Spring MVC 和 请求数据
java·spring·mvc
DianSan_ERP2 小时前
架构师视角:电商大促高并发下的订单API限流与防漏单架构演进
java·运维·网络·安全·微服务·架构·自动化
云烟成雨TD2 小时前
Agent Scope Java 2.x 系列【6】消息层
java·人工智能·agent
云烟成雨TD2 小时前
Spring AI Alibaba 1.x 系列【74】Agentic RAG 与混合 RAG
java·人工智能·spring