springboot启动的整体流程解析
在SpringApplication初始化(详见:springboot启动源码解析(一):SpringApplication初始化)之后,开始了真正意义上的启动过程,它的整个启动过程在SpringApplication的run()方法中。
public static ConfigurableApplicationContext run(Class<?>[] primarySources, String[] args) {
return new SpringApplication(primarySources).run(args);
}
public ConfigurableApplicationContext run(String... args) {
// 首先初始化一个计时器,并开始了启动计时
StopWatch stopWatch = new StopWatch();
stopWatch.start();
// 初始化一个启动上下文
DefaultBootstrapContext bootstrapContext = createBootstrapContext();
ConfigurableApplicationContext context = null;
// 这一步是为了设置系统参数:java.awt.headless(在系统可能缺少显示设备、键盘或鼠标这些外设的情况下可以使用该模式,例如Linux服务器)
configureHeadlessProperty();
// listeners中维护着一个监听器列表,它们贯穿了整个启动过程,负责在各个启动步骤完成后发布相应事件事
// listeners中维护的监听器列表是通过调用getSpringFactoriesInstances方法,从spring.factories中获取得到的类型为SpringApplicationRunListener的实例列表,但此列表中有且仅有一个事件发布监听器:EventPublishingRunListener
SpringApplicationRunListeners listeners = getRunListeners(args);
// 此处发布springboot开始启动的事件
listeners.starting(bootstrapContext, this.mainApplicationClass);
try {
// 初始化应用参数,此参数从启动命令中进行解析,例如java -jar --spring.profiles.active=prod等
ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);
// 根据运行监听器和应用参数来准备spring环境
ConfigurableEnvironment environment = prepareEnvironment(listeners, bootstrapContext, applicationArguments);
// 配置需要忽略的Bean信息,该信息从应用环境中"spring.beaninfo.ignore"属性取得,如果未取到,则将默认值TRUE设置到System的"spring.beaninfo.ignore"属性中
configureIgnoreBeanInfo(environment);
// 获取并打印banner,并返回生成的PrintedBanner实例
Banner printedBanner = printBanner(environment);
// 创建应用上下文
context = createApplicationContext();
// 为此应用上下文设置applicationStartup,使应用程序上下文在启动期间能够记录一些指标,applicationStartup = ApplicationStartup.DEFAULT
context.setApplicationStartup(this.applicationStartup);
// 准备应用程序上下文,该步骤包含一个非常关键的操作:将启动类注入容器,为后续开启自动化提供基础
prepareContext(bootstrapContext, context, environment, listeners, applicationArguments, printedBanner);
// 刷新应用上下文,在此步骤中,进行了类扫描,bean注入等关键操作
refreshContext(context);
// 做一些刷新应用上下文的后序处理操作
afterRefresh(context, applicationArguments);
// 停止计时监控
stopWatch.stop();
// 输出日志记录执行主类名、时间信息
if (this.logStartupInfo) {
new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), stopWatch);
}
// 发布应用上下文启动事件
listeners.started(context);
// 执行所有的Runner运行器
callRunners(context, applicationArguments);
}
catch (Throwable ex) {
handleRunFailure(context, ex, listeners);
throw new IllegalStateException(ex);
}
try {
// 发布应用上下文就绪事件
listeners.running(context);
}
catch (Throwable ex) {
handleRunFailure(context, ex, null);
throw new IllegalStateException(ex);
}
// 返回应用上下文
return context;
}