SpringApplication 这个类应该算是 Spring Boot 框架 本身的“创新”产物了,因为原始的Spring框架中并没有这个类,SpringApplication 里面封装了一套Spring应用的启动流程,然而这对用户完全透明,因此我们上手 Spring Boot 时感觉简洁且轻量。
一般来说默认的 SpringApplication 执行流程已经可以满足大部分需求,但是 若用户想干预这个过程,则可以通过 SpringApplication 在流程某些地方开启的 扩展点 来完成对流程的扩展,典型的扩展方案那就是使用 set 方法。
我们来举一个栗子,把我们天天司空见惯的 Spring Boot 应用的启动类来拆解一下写出来:
@SpringBootApplication
public class CodeSheepApplication {
public static void main( String[] args ) {
// SpringApplication.run( CodeSheepApplication.class args ); // 这是传统Spring Boot应用的启动,一行代码搞定,内部默认做了很多事
SpringApplication app = new SpringApplication( CodeSheepApplication.class );
app.setXXX( ... ); // 用户自定的扩展在此 !!!
app.run( args );
}
}
这样一拆解后我们发现,我们也需要先构造 SpringApplication 类对象,然后调用该对象的 run() 方法。
那么接下来就聊聊 SpringApplication 的构造过程 以及其 run()方法的流程,搞清楚了这个,那么也就搞清楚了Spring Boot应用是如何运行起来的了。
SpringApplication实例的初始化
还是先对照代码来看:
四个关键的步骤已标注在图中,分别解释如下:
① 推断应用的类型:创建的是REACTIVE应用、SERVLET应用、NONE 三种中的某一种
② 使用
SpringFactoriesLoader查找并加载classpath下META-INF/spring.factories文件中所有可用的ApplicationContextInitializer
③ 使用
SpringFactoriesLoader查找并加载classpath下META-INF/spring.factories文件中的所有可用的ApplicationListener
④ 推断并设置
main方法的定义类
SpringApplication的run()方法探秘
先看看代码长啥样子:
关键步骤都已经用数字标注在上图之中了,除此之外,这里也画了一个流程图对照理解:
我们将各步骤总结精炼如下:
通过
SpringFactoriesLoader加载META-INF/spring.factories文件,获取并创建SpringApplicationRunListener对象然后由
SpringApplicationRunListener来发出starting消息创建参数,并配置当前
SpringBoot应用将要使用的Environment完成之后,依然由
SpringApplicationRunListener来发出environmentPrepared消息创建
ApplicationContext初始化
ApplicationContext,并设置Environment,加载相关配置等由
SpringApplicationRunListener来发出contextPrepared消息,告知Spring Boot 应用使用的ApplicationContext已准备OK将各种
beans装载入ApplicationContext,继续由SpringApplicationRunListener来发出contextLoaded消息,告知 Spring Boot 应用使用的ApplicationContext已装填OKrefresh ApplicationContext,完成IoC容器可用的最后一步由
SpringApplicationRunListener来发出started消息调用
callRunners(...)方法,让实现了ApplicationRunner和CommandLineRunner接口类的run方法得以执行,用于在 Spring 应用上下文准备完毕后,执行一些额外操作。从而完成最终的程序的启动。由
SpringApplicationRunListener来发出running消息,告知程序已运行起来了