过早客
  • 首页
  • 节点
  • 成员
  • 广告投放
  • 登录
  • 注册

SpringBoot 应用程序启动过程探秘

IT技术 • CodeSheep • 发表于 6 年前 • 最后回复来自 heygrl • 6 年前

Profile

本文共 946字,阅读大约需要 3分钟 !


概述

说到接触 SpringBoot 伊始,给我第一映像最深的是有两个关键元素:

SpringBoot 的两个关键元素

对照上面的典型代码,这个两个元素分别是:

  • @SpringBootApplication
  • SpringApplication 以及 run() 方法

关于 @SpringBootApplication 注解的剖析已经在上文:《SpringBoot 中 @SpringBootApplication注解背后的三体结构探秘》 中完成了,其实它背后就是一个三体结构,只是 SpringBoot给了其一个包装而已。那么本文我们就来看看这个 SpringApplication 以及 run() 方法 到底是个什么鬼,它背后又隐藏了哪些奥秘呢?

注: 本文首发于 My Personal Blog,欢迎光临 小站

本文内容脑图如下:

本文内容脑图


SpringApplication 惊鸿一瞥

SpringApplication 这个类应该算是 SpringBoot 框架 的“创新”产物了,原始的 Spring中并没有这个类,SpringApplication 里面封装了一套 Spring 应用的启动流程,然而这对用户完全透明,因此我们上手 SpringBoot 时感觉简洁、轻量。

一般来说默认的 SpringApplication 执行流程已经可以满足大部分需求,但是 若用户想干预这个过程,则可以通过 SpringApplication 在流程某些地方开启的 扩展点 来完成对流程的扩展,典型的扩展方案那就是使用 set 方法。

我们来举一个栗子,把我们天天司空见惯的 SpringBoot 应用的启动类来拆解一下写出来:

@SpringBootApplication
public class CodeSheepApplication {
    public static void main( String[] args ) {
        // SpringApplication.run( CodeSheepApplication.class args ); // 这是传统SpringBoot应用的启动,一行代码搞定,内部默认做了很多事
        SpringApplication app = new SpringApplication( CodeSheepApplication.class );
        app.setXXX( ... ); // 用户自定的扩展在此 !!!
        app.run( args );
    }
}

这样一拆解后我们发现,我们也需要先构造 SpringApplication 类对象,然后调用该对象的 run() 方法。那么接下来就讲讲 SpringApplication 的构造过程 以及其 run() 方法的流程,搞清楚了这个,那么也就搞清楚了SpringBoot应用是如何运行起来的!


SpringApplication 实例的初始化

我们对照代码来看:

SpringApplication 实例的初始化代码

四个关键的步骤已标注在图中,分别解释如下:

  • ① 推断应用的类型:创建的是 REACTIVE应用、SERVLET应用、NONE 三种中的某一种

推断应用的类型

  • ② 使用 SpringFactoriesLoader查找并加载 classpath下 META-INF/spring.factories文件中所有可用的 ApplicationContextInitializer

ApplicationContextInitializer 加载

  • ③ 使用 SpringFactoriesLoader查找并加载 classpath下 META-INF/spring.factories文件中的所有可用的 ApplicationListener

ApplicationListener 加载

  • ④ 推断并设置 main方法的定义类

推断并设置main方法的定义类



SpringApplication 的run()方法探秘

先看看代码长啥样子:

SpringApplication 的run()方法背后的奥秘

各个主要步骤我已经标注在上图之中了,除此之外,我也按照自己的理解画了一个流程图如下所示,可以对照数字标示看一下:

SpringBoot 应用启动流程图

我们将各步骤总结精炼如下:

  1. 通过 SpringFactoriesLoader 加载 META-INF/spring.factories 文件,获取并创建 SpringApplicationRunListener 对象

  2. 然后由 SpringApplicationRunListener 来发出 starting 消息

  3. 创建参数,并配置当前 SpringBoot 应用将要使用的 Environment

  4. 完成之后,依然由 SpringApplicationRunListener 来发出 environmentPrepared 消息

  5. 创建 ApplicationContext

  6. 初始化 ApplicationContext,并设置 Environment,加载相关配置等

  7. 由 SpringApplicationRunListener 来发出 contextPrepared 消息,告知SpringBoot 应用使用的 ApplicationContext 已准备OK

  8. 将各种 beans 装载入 ApplicationContext,继续由 SpringApplicationRunListener 来发出 contextLoaded 消息,告知 SpringBoot 应用使用的 ApplicationContext 已装填OK

  9. refresh ApplicationContext,完成IoC容器可用的最后一步

  10. 由 SpringApplicationRunListener 来发出 started 消息

  11. 完成最终的程序的启动

  12. 由 SpringApplicationRunListener 来发出 running 消息,告知程序已运行起来了

至此,全流程结束!



后记

由于能力有限,若有错误或者不当之处,还请大家批评指正,一起学习交流!

  • My Personal Blog
  • 我的半年技术博客之路


可 长按 或 扫描 下面的 小心心 来订阅作者公众号 CodeSheep,获取更多 务实、能看懂、可复现的 原创文 ↓↓↓

CodeSheep · 程序羊


加入收藏 新浪微博 分享到微信 ❤赞 1132 次点击 2 人赞 1 人收藏

打开微信“扫一扫”,打开网页后点击屏幕右上角分享按钮

共收到1条回复
heygrl 6 年前 #1 赞 0

赞,最近也在学习springboot

请绑定手机号后,再发言,点击此处
Guozaoke.com—源自武汉的高端交流分享社区
相关主题
求推荐一个AI智能体客服
Android手机推荐
有多少人知道华为在AI算力的基础建设上已经超越英伟达了?
请教机器学习人工智能的一个技术问题
搞了个AI 生图的网站,不需要登录,永久免费
[第二波送码]动动嘴皮,AI秒出图!2025最炸裂图标神器:免费生成+手机实时预览
我没房子的时候,每年赚的钱比不上房子每年涨的钱。等我咬牙买了,每年赚的钱,比不上房子每年亏的钱
开发了一个管理 Linux 服务器的桌面可视化管理工具,求蹂躏
社区里有没有对钢架自行车和钢架builder感兴趣的同志
社友们,车险异地投保有了解的吗

过早客微信公众号:guozaoke • 过早客新浪微博:@过早客 • 广告投放合作微信:fullygroup50 鄂ICP备2021016276号-2 • 鄂公网安备42018502001446号