介绍一下Spring Boot启动流程以及生命周期勾子函数?
代码那些事儿 2020-02-22 16:10:15
上篇文章介绍了Spring Boot自动配置的原理,但是对于Spring Boot应用的整个启动流程还是一头雾水,虽然Spring Boot大大简化了我们的代码配置工作,但是对于内部原理我们还是需要了解一下的。本篇文章我们主要对照源码介绍一下应用启动入口SpringApplication.run()方法的整个执行流程。
SpringApplication构造函数
首先我们进到run()方法里,这里可以看到构造了一个SpringApplication对象,继续进构造方法看一下
构造方法里主要做了四件事:
- 根据classpath下是否存在某些特定类来确定应用的类型,包括:REACTIVE、SERVLET和NONE三种类型,其中REACTIVE、SERVLET会以WEB应用的形式启动,NONE则会在应用执行完毕后结束进程
- 利用SpringFactoriesLoader自动加载spring.factories文件中配置的 ApplicationContextInitializer,ApplicationContextInitializer原本是Spring框架Context模块下的一个接口,Spring Boot内部提供了一些该接口的实现类,使得我们有机会对ApplicationContext做一些特定的初始化工作
- 利用SpringFactoriesLoader自动加载spring.factories文件中配置的 ApplicationListener,ApplicationListener同样是Context模块下的接口,Spring Boot内部提供了一些该接口的实现类,使得我们有机会在监听到某些应用事件发生时做一些特定的工作。
- 根据main方法执行堆栈确定应用的主引导类,即Spring Boot项目自带的XXApplication类
run方法
对照应用启动的整个生命周期回调函数,run方法内部大致可分为六个主要步骤:
- 利用SpringFactoriesLoader自动加载spring.factories文件中配置的SpringApplicationRunListener后,回调starting方法,表示run()方法开始执行
- 根据配置文件及参数构建Environment结束时,回调environmentPrepared方法,表示应用的Environment已准备好
- 打印banner,创建ApplicationContext,设置好Environment并初始化结束时,回调contextPrepared方法,表示应用的ApplicationContext已准备好
- 在ApplicationContext将所有已经扫描到的beans加载完成时,回调contextLoaded方法,表示ApplicationContext 已经加载完成
- 当ApplicationContext刷新结束并完成后续启动后,回调started方法,表示应用已经启动
- 遍历执行应用所有的CommandLineRunner结束后,回调running方法,表示应用正在运行
控制台输出
分析了这么多,最后我们启动应用看一下控制台输出是不是上面这个流程。
想要监听到Spring Boot应用启动生命周期的回调,首先我们需要自定义一个SpringApplicationRunListener接口实现类:
同时,我们在resources目录下创建spring.factories文件:
接下来我们启动应用,等待控制台输出:
通过与上面run方法执行流程对比,可以确定Spring Boot应用启动过程与我们分析的基本一致。