SpringBoot 的启动引导类真的是 XXApplication 吗?

1. 引导类的“名”与“实”:为什么大家都叫 Application?

Spring Boot 提倡“约定优于配置”的理念。XXXApplication 这个命名方式,本质上是官方文档和绝大多数教程为了统一、规范和易于识别而推广的一种约定。它就像是一个显眼的“入口”路牌,告诉所有开发者:“嘿,项目的起点就在这里!”

所以,回答你的第一个问题:“真的是 XXApplication 吗?”——在约定上,是的;但在语法上,不是。 它的真实身份,是一个被 @SpringBootApplication 注解标记的、包含 main 方法的普通 Java 类。

2. 拆解引导类的“灵魂”:@SpringBootApplication

引导类的强大,并非源自它的名字,而是它头顶的那个神奇的注解——@SpringBootApplication。这个注解本身并不复杂,但它是一个“三合一”的组合注解,是开启 Spring Boot 世界的钥匙。

java

// 一个典型的Spring Boot引导类 @SpringBootApplication // <-- 核心中的核心 public class MyAwesomeApp { public static void main(String[] args) { SpringApplication.run(MyAwesomeApp.class, args); } }

让我们一层层剥开 @SpringBootApplication 的外衣 :

  • @SpringBootConfiguration:它本身又是一个 @Configuration 注解的“马甲”。这意味着,我们的引导类本身就是一个配置类。你可以在这个类里通过 @Bean 注解向 Spring 容器注册额外的 Bean。
  • @ComponentScan:这就是 “组件扫描” 注解。它的默认扫描范围是引导类所在包及其所有子包 。这就是为什么我们通常建议将引导类放在项目的根目录下,比如 com.example.myproject,这样它就能自动扫描到 com.example.myproject.controllercom.example.myproject.service 等子包下的 @Controller@Service 等组件。
  • @EnableAutoConfiguration:这是整个 Spring Boot 框架的灵魂所在,负责开启自动配置。它会根据你在项目类路径中添加的依赖(比如 spring-boot-starter-web),自动推断并配置你需要的 Bean。例如,你添加了 web starter,它就会自动帮你配置好 DispatcherServlet 和内嵌的 Tomcat 服务器。

3. 深度剖析:SpringApplication.run() 里到底发生了什么?

引导类的 main 方法只做了一件事:委托给 SpringApplication.run() 静态方法。但这一行代码背后,是一套极其精巧的启动流程。整个过程可以分为两大阶段:初始化 和 运行 。

3.1 初始化阶段:准备舞台

当调用 run 方法时,会先创建一个 SpringApplication 实例。在这个构造函数中,Spring Boot 会完成一系列“侦查”工作:

  1. 推断应用类型:检查类路径下是存在 DispatcherServlet(传统 Servlet)还是 DispatcherHandler(WebFlux),从而确定应用类型是响应式 Web 还是 Servlet Web,抑或是非 Web 应用 。
  2. 加载初始化器:从 META-INF/spring.factories 文件中加载所有配置的 ApplicationContextInitializer。这些初始化器用于在 Spring 容器刷新前对其进行定制。
  3. 加载监听器:同样从 spring.factories 中加载所有配置的 ApplicationListener。这些监听器会监听启动过程中的不同阶段事件,并执行相应动作 。
  4. 推断主启动类:通过堆栈信息找到含有 main 方法的类,也就是我们的引导类。
3.2 运行阶段:正式开演

初始化完成后,真正的 run() 逻辑开始执行,整个过程充满了事件驱动 。

  • ApplicationStartingEvent:发布应用启动事件。
  • 准备环境 prepareEnvironment():创建 Environment 对象,并加载系统环境变量、JVM 参数以及最重要的 application.properties/yml 配置文件。完成后发布 ApplicationEnvironmentPreparedEvent 。
  • 创建上下文 createApplicationContext():根据初始化阶段推断的应用类型,创建对应的 ApplicationContext(IoC 容器)。例如,对于 Servlet Web 应用,会创建 AnnotationConfigServletWebServerApplicationContext 。
  • 准备上下文 prepareContext():将 Environment 绑定到上下文,执行所有已加载的 ApplicationContextInitializer,并通过 @ComponentScan 将引导类本身作为一个配置源,将其 Bean 定义加载到容器中。之后发布 ApplicationPreparedEvent
  • 刷新上下文 refreshContext():这是 Spring 框架最核心的方法。它会完成所有 Bean 的实例化、依赖注入、初始化,以及最重要的——触发自动配置。Tomcat 等内嵌 Web 服务器也是在此步骤中启动的 。
  • 启动后动作 afterRefresh():执行所有实现了 ApplicationRunner 和 CommandLineRunner 接口的回调类,用于执行启动后的初始化任务 。
  • ApplicationReadyEvent:发布应用已准备就绪事件,应用正式对外提供服务。

4. 进阶探索:引导类还可以这样玩

除了最简单的 SpringApplication.run(),我们还可以通过创建和定制 SpringApplication 对象来实现更精细的控制 。

java

@SpringBootApplication public class CustomizedApp { public static void main(String[] args) { // SpringApplication app = new SpringApplication(CustomizedApp.class); SpringApplication app = new SpringApplication(); app.setPrimarySources(Set.of(CustomizedApp.class)); app.setBannerMode(Banner.Mode.OFF); // 关闭Banner app.setDefaultProperties(Collections .singletonMap("server.port", "8081")); // 设置默认端口 // 添加自定义监听器 app.addListeners(new MyApplicationListener()); ConfigurableApplicationContext context = app.run(args); // ... 可以获取context做一些事情 } }

5. 终极揭秘:Jar 包启动的“障眼法”

当我们把 Spring Boot 项目打成可执行 Jar 包并运行(java -jar myapp.jar)时,引导类的角色变得更加有趣。你可能会想,我明明设置了 MyAwesomeApp 作为主类,为什么启动时还能看到 Spring Boot 的 Banner?

答案是,真正的“引导”另有其人。Spring Boot 的 Jar 包结构是特殊的,它的 META-INF/MANIFEST.MF 文件里定义了真实的入口 :

properties

Main-Class: org.springframework.boot.loader.launch.JarLauncher Start-Class: com.example.myproject.MyAwesomeApp

  • Main-Class:这里配置的是 JarLauncher,它是 Spring Boot 自己提供的引导程序。它的工作是创建自定义的类加载器,从 Jar 包内部的 BOOT-INF/lib 目录下加载所有依赖的 Jar 包。
  • Start-Class:这里才指向我们编写的引导类 MyAwesomeAppJarLauncher 在准备好类加载器后,会通过反射调用 MyAwesomeApp 的 main 方法,从而正式启动我们的 Spring Boot 应用 。

所以,从物理 Jar 包层面看,真正的“引导类”是 JarLauncher;而从业务逻辑层面看,MyAwesomeApp 仍然是当之无愧的“启动引导类”。这是一种巧妙的分层设计。

总结

现在我们可以清晰地回答你的问题了:

  1. 命名约定XXApplication 是官方推荐的命名习惯,但不是强制规定。
  2. 本质核心:引导类的本质是一个被 @SpringBootApplication 注解标记的配置类,它作为组件扫描和自动配置的起点。
  3. 启动原理:引导类通过 SpringApplication.run() 触发一系列复杂的初始化、事件发布和容器刷新流程,最终启动内嵌服务器,完成应用自举。
  4. 终极真相:在打包成可执行 Jar 后,物理上的引导类是 JarLauncher,它负责创建环境并调用我们定义的业务引导类。

Read more

VsCode远程连接服务器后安装Github Copilot无法使用

VsCode远程连接服务器后安装Github Copilot无法使用

VsCode远程连接服务器后安装Github Copilot无法使用 1.在Vscode的settings中搜索Extension Kind,如图所示: 2.点击Edit in settings.json,添加如下代码: "remote.extensionKind":{"GitHub.copilot":["ui"],"GitHub.copilot-chat":["ui"],} remote.extensionKind 的作用 这是 VS Code 的远程开发配置项,用于控制扩展在远程环境(如 SSH、容器、WSL)中的运行位置。可选值: “ui”:扩展在本地客户端运行 “workspace”:扩展在远程服务器运行 这两个扩展始终在 本地客户端运行,

By Ne0inhk
手动部署开源OpenClaw汉化中文版过程中常见问题排查手册

手动部署开源OpenClaw汉化中文版过程中常见问题排查手册

部署开源OpenClaw汉化中文版过程中常见问题排查手册 遇到问题?按错误消息搜索本页,或按场景分类查找解决方案。 🎬 摘要 😤 部署 OpenClaw 汉化版又双叒叕报错了? 别慌!这份实战排查手册专为「踩坑」而生。 无论是 Docker 镜像拉取失败、容器启动闪退,还是 Dashboard 死活连不上、远程访问 502 报错——我们按错误场景分类整理,支持按错误关键词秒搜定位。每个解决方案均来自真实部署案例,附带紧急修复通道和根因分析,让你从「报错一脸懵」到「秒级排障」。 🔧 适用版本:OpenClaw 汉化中文版(Docker 部署) 📌 更新策略:与主仓库每小时同步,排查方案持续迭代 ⚡ 建议收藏:部署前通读「零、紧急修复」,关键时刻能救命! 目录 * 零、紧急修复 ⚠️ * 一、安装问题 * 二、启动问题

By Ne0inhk
Flutter 三方库 github_actions_toolkit 的鸿蒙化适配指南 - 实现 GitHub Actions 高效自动化任务构建、支持日志颜色修饰与核心工具集成

Flutter 三方库 github_actions_toolkit 的鸿蒙化适配指南 - 实现 GitHub Actions 高效自动化任务构建、支持日志颜色修饰与核心工具集成

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 github_actions_toolkit 的鸿蒙化适配指南 - 实现 GitHub Actions 高效自动化任务构建、支持日志颜色修饰与核心工具集成 前言 在进行 Flutter for OpenHarmony 的工程化 CI/CD(持续集成与交付)构建时,利用 GitHub Actions 进行自动化测试和流水线发布是主流选择。github_actions_toolkit 是一个专为编写非 Web 类 Action 脚本设计的工具集,它能让你在 Dart 脚本中轻松调用 Actions 的核心功能(如日志分级输出、设置导出变量等)。本文将探讨如何利用该库提升鸿蒙项目的自动化构建效率。 一、原理解析 / 概念介绍

By Ne0inhk
GitHub热榜----上帝视角玩转未来!MiroFish:基于群体智能的万物预测引擎

GitHub热榜----上帝视角玩转未来!MiroFish:基于群体智能的万物预测引擎

摘要:你是否想过像《黑客帝国》或《西部世界》那样,构建一个平行的数字世界?或者在小说写到瓶颈时,让书中人物自己“活”过来推演结局?今天介绍的开源项目 MiroFish,正是一个基于**多智能体(Multi-Agent)**技术的通用群体智能引擎。它能通过你上传的“种子信息”,自动生成成千上万个具备独立人格和记忆的智能体,在数字沙盘中演化未来。 🚀 前言:当 AI 拥有了“社会属性” 在 ChatGPT 单打独斗的时代,我们问它:“如果发生X,会产生什么后果?”它只能基于训练数据给出概率性的回答。 但在 MiroFish 构建的多智能体系统 (MAS) 中,AI 不再是一个孤独的对话框。MiroFish 让无数个 AI 智能体组成一个社会,它们有记忆、有性格、有社交关系。当你在系统中投入一个变量(比如一条突发新闻),你会看到这些智能体如何反应、

By Ne0inhk