启动过程结论

  • 推测web应用类型。
  • spi的方式获取BootstrapRegistryInitializer、ApplicationContextInitializer、ApplicationContextInitializer对象。
  • 通过调用栈推测出main()方法所在的类。
  • 调用启动方法:run(String... args)。后面的步骤在这个方法内部!
  • 触发SpringApplicationRunListener的starting()。
  • 创建Environment对象。
  • 触发SpringApplicationRunListener的environmentPrepared()。
  • 打印Banner。
  • 创建Spring容器对象(ApplicationContext)。
  • 利用ApplicationContextInitializer初始化Spring容器对象。
  • 触发SpringApplicationRunListener的contextPrepared()。
  • 调用DefaultBootstrapContext对象的close()。
  • 将启动类作为配置类注册到Spring容器中(load()方法)。
  • 触发SpringApplicationRunListener的contextLoaded()。
  • 刷新Spring容器。
  • 触发SpringApplicationRunListener的started()。
  • 调用ApplicationRunner和CommandLineRunner。
  • 触发SpringApplicationRunListener的ready()。
  • 上述过程抛异常了就触发SpringApplicationRunListener的failed()。

入口位置

  • 第一步构造对象:new SpringApplication(primarySources)
  • 第二步调用SpringApplication.run(String... args)
// 我们自己写的main方法
@SpringBootApplication
public class ZfcqApp {
public static void main(String[] args) {
SpringApplication.run(ZfcqApp.class, args);
}
} /**
* 他会调用SpringApplication的run方法
* primarySource:我们传入的类的class
* args:我们传入的参数,一般是启动的时候-D制定的参数
*/
public static ConfigurableApplicationContext run(Class<?> primarySource, String... args) {
return run(new Class<?>[] { primarySource }, args);
} /**
* 继续看内部调用的run方法
*/
public static ConfigurableApplicationContext run(Class<?>[] primarySources, String[] args) {
// 俩步,第一步构造对象:new SpringApplication(primarySources)
// 第二步调用SpringApplication.run(String... args)
// 这里我们可以在main方法中分开俩步去写,从而可以在中间设置SpringApplication对象的信息
return new SpringApplication(primarySources).run(args);
}

构造SpringApplication源码分析

/**
* 调用俩个参数的构造方法
* primarySources:我们主类的class
*/
public SpringApplication(Class<?>... primarySources) {
this(null, primarySources);
} /**
* 俩个产生的构造方法
*/
public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {
// 赋值到全局,这里第一次传入的是null
this.resourceLoader = resourceLoader;
// 主类存在的判断
Assert.notNull(primarySources, "PrimarySources must not be null");
// 赋值到全局,这里一般传入的是我们的main方法的类
this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources)); // 推测WEB应用的类型(NONE、SERVLET、REACTIVE)
this.webApplicationType = WebApplicationType.deduceFromClasspath(); // 从spring.factories中获取BootstrapRegistryInitializer对象的值
this.bootstrapRegistryInitializers = new ArrayList<>(getSpringFactoriesInstances(BootstrapRegistryInitializer.class)); // 从spring.factories中获取ApplicationContextInitializer对象的值
setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class)); // 从spring.factories中获取ApplicationListener对象的值。非常重要的有一个是EnvironmentPostProcessorApplicationListener
setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class)); // 推测出main方法所在的类!SpringApplication.run(ZfcqApp.class, args);可以传任意的类!
this.mainApplicationClass = deduceMainApplicationClass();
}

推测web应用类型

  • REACTIVE:web应用。
  • NONE:无Servlet,不是web应用。
  • SERVLET:除去上面俩种的其他应用。
static WebApplicationType deduceFromClasspath() {
// 如果项目依赖中存在org.springframework.web.reactive.DispatcherHandler,并且不存在org.springframework.web.servlet.DispatcherServlet,那么应用类型为WebApplicationType.REACTIVE
if (ClassUtils.isPresent(WEBFLUX_INDICATOR_CLASS, null) && !ClassUtils.isPresent(WEBMVC_INDICATOR_CLASS, null) && !ClassUtils.isPresent(JERSEY_INDICATOR_CLASS, null)) {
return WebApplicationType.REACTIVE;
}
// 如果项目依赖中不存在org.springframework.web.reactive.Dispatche#rHandler,也不存在org.springframework.web.servlet.DispatcherServlet,那么应用类型为WebApplicationType.NONE
for (String className : SERVLET_INDICATOR_CLASSES) {
if (!ClassUtils.isPresent(className, null)) {
return WebApplicationType.NONE;
}
}
// 否则,应用类型为WebApplicationType.SERVLET
return WebApplicationType.SERVLET;
}

推测出Main类(main()方法所在的类)

  • 从调用栈中去获取main类!
private Class<?> deduceMainApplicationClass() {
try {
// 获取调用栈数组
StackTraceElement[] stackTrace = new RuntimeException().getStackTrace();
// 找到调用栈中的main方法!
for (StackTraceElement stackTraceElement : stackTrace) {
if ("main".equals(stackTraceElement.getMethodName())) {
return Class.forName(stackTraceElement.getClassName());
}
}
}
catch (ClassNotFoundException ex) {
// Swallow and continue
}
return null;
}

核心的启动方法:run(String... args)

public ConfigurableApplicationContext run(String... args) {
// 开始时间
long startTime = System.nanoTime(); // 创建引导启动器,类似一个ApplicationContext,可以往里面添加一些对象。
DefaultBootstrapContext bootstrapContext = createBootstrapContext(); // spring容器对象,开始的时候为空
ConfigurableApplicationContext context = null;
// 与awt有关,一般用不上
configureHeadlessProperty(); // spring boot的启动监听器
// 从spring.factories中获取SpringApplicationRunListener对象的值
// 默认会拿到EventPublishingRunListener,他会启动各个地方的ApplicationEvent事件。
SpringApplicationRunListeners listeners = getRunListeners(args); // 发布开始启动的事件:ApplicationstartingEvent
listeners.starting(bootstrapContext, this.mainApplicationClass);
try {
// 把run方法的参数进行封装
ApplicationArguments applicationArguments = new DefaultApplicationArguments(args); // 准备environment对象:包括操作系统、jvm、ymal、properties....配置
// 发布一个ApplicationEnvironmentPreparedEvent事件。
ConfigurableEnvironment environment = prepareEnvironment(listeners, bootstrapContext, applicationArguments); // 默认spring.beaninfo.ignore为true,表示不需要jdk去缓存BeanInfo信息,spring自己缓存。
// 这里是spring在创建bean的时候会利用jdk的一些工具来解析一个类的相关信息,jdk在解析一个类的信息的时候会进行缓存,这里就是禁止了jdk的缓存。
configureIgnoreBeanInfo(environment); // 打印Banner
Banner printedBanner = printBanner(environment); // 根据应用类型,创建spring容器
context = createApplicationContext();
// jdk9的一个机制,默认没做任何操作
context.setApplicationStartup(this.applicationStartup); // 准备容器的操作
// 利用ApplicationContextInitializer初始化spring容器
// 发布ApplicationContextInitializedEvent(容器初始化完成)事件
// 发布BootstrapContextClosedEvent(关闭引导容器)事件
// 注册primarySources类(run方法存进来的配置类)
// 发布ApplicationPreparedEvent事件
prepareContext(bootstrapContext, context, environment, listeners, applicationArguments, printedBanner); // refresh容器:解析配置类、扫描、启动webServer。spring相关的逻辑!
refreshContext(context); // 空方法,类似spring的onRefresh方法,可以由子类实现。
afterRefresh(context, applicationArguments); // 启动的事件
Duration timeTakenToStartup = Duration.ofNanos(System.nanoTime() - startTime);
if (this.logStartupInfo) {
new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), timeTakenToStartup);
} // 发布一个ApplicationStartedEvent事件,表示spring已经启动完成
listeners.started(context, timeTakenToStartup); // 从spring容器中获取ApplicationRunner和CommandLineRunner,并执行他的run方法。
// 这俩个可以自己定义Bean
callRunners(context, applicationArguments);
}
catch (Throwable ex) {
// 失败之后,发布一个失败的事件:ApplicationFailedEvent
handleRunFailure(context, ex, listeners);
throw new IllegalStateException(ex);
}
try {
// 计算下事件
Duration timeTakenToReady = Duration.ofNanos(System.nanoTime() - startTime); // 一切都成功,发布ApplicationStartedEvent事件
listeners.ready(context, timeTakenToReady);
}
catch (Throwable ex) {
// 失败之后,发布一个失败的事件:ApplicationFailedEvent
handleRunFailure(context, ex, null);
throw new IllegalStateException(ex);
}
return context;
}

创建引导启动器

private DefaultBootstrapContext createBootstrapContext() {
// 构建一个DefaultBootstrapContext对象,这个对象是2.4.0之后才会有!
DefaultBootstrapContext bootstrapContext = new DefaultBootstrapContext();
// 利用bootstrapRegistryInitializers初始化bootstrapContext。可以在spring.factories中设置bootstrapRegistryInitializers
this.bootstrapRegistryInitializers.forEach((initializer) -> initializer.initialize(bootstrapContext));
return bootstrapContext;
}

准备environment

private ConfigurableEnvironment prepareEnvironment(SpringApplicationRunListeners listeners, DefaultBootstrapContext bootstrapContext, ApplicationArguments applicationArguments) {
// Create and configure the environment 这句话是源码中本身的注释!
// 创建ApplicationServletEnvironment。
ConfigurableEnvironment environment = getOrCreateEnvironment(); // 添加SimpleCommandLinePropertySource放在首位
configureEnvironment(environment, applicationArguments.getSourceArgs()); // 把所有的PropertySource封装为ConfigurationPropertySourcesPropertySource,然后添加到environment中,放在首位!
ConfigurationPropertySources.attach(environment); // 发布ApplicationEnvironmentPreparedEvent(应用Environment准备完成)事件,表示环境已经准备好了。默认EnvironmentPostProcessorApplicationListener去处理这个事件!
listeners.environmentPrepared(bootstrapContext, environment); // 把DefaultProperties放到最后
DefaultPropertiesPropertySource.moveToEnd(environment); // 环境中spring.main.environment-prefix参数校验
Assert.state(!environment.containsProperty("spring.main.environment-prefix"), "Environment prefix cannot be set via properties."); // 环境中spring.main参数校验
bindToSpringApplication(environment);
if (!this.isCustomEnvironment) {
environment = convertEnvironment(environment);
}
ConfigurationPropertySources.attach(environment);
return environment;
}

准备spring容器的操作

private void prepareContext(DefaultBootstrapContext bootstrapContext, ConfigurableApplicationContext context, ConfigurableEnvironment environment, SpringApplicationRunListeners listeners, ApplicationArguments applicationArguments, Banner printedBanner) {
// 设置环境变量到spring容器
context.setEnvironment(environment); // 设置在SpringApplication上的BeanNameGenerator、resourceLoader设置到spring容器中
postProcessApplicationContext(context); // 使用ApplicationContextInitializer初始化spring容器
applyInitializers(context); // 容器初始化完成,发布ApplicationContextInitializedEvent事件
listeners.contextPrepared(context); // 关闭引导的容器
bootstrapContext.close(context); if (this.logStartupInfo) {
logStartupInfo(context.getParent() == null);
logStartupProfileInfo(context);
} // 注册一些单例Bean
// Add boot specific singleton beans
ConfigurableListableBeanFactory beanFactory = context.getBeanFactory();
beanFactory.registerSingleton("springApplicationArguments", applicationArguments);
if (printedBanner != null) {
beanFactory.registerSingleton("springBootBanner", printedBanner);
} // spring容器设置AllowCircularReferences和allowBeanDefinitionOverriding
if (beanFactory instanceof AbstractAutowireCapableBeanFactory) {
((AbstractAutowireCapableBeanFactory) beanFactory).setAllowCircularReferences(this.allowCircularReferences);
if (beanFactory instanceof DefaultListableBeanFactory) {
((DefaultListableBeanFactory) beanFactory) .setAllowBeanDefinitionOverriding(this.allowBeanDefinitionOverriding);
}
}
if (this.lazyInitialization) {
context.addBeanFactoryPostProcessor(new LazyInitializationBeanFactoryPostProcessor());
}
// Load the sources
// 拿到启动配置(run方法传递进来的)
Set<Object> sources = getAllSources();
Assert.notEmpty(sources, "Sources must not be empty");
// 将启动配置类解析为BeanDefinition注册到spring容器
load(context, sources.toArray(new Object[0]));
// 发布ApplicationPreparedEvent事件,表示已经启动好spring容器
listeners.contextLoaded(context);
}

结束语

  • 你的点赞是我提高文章质量最大的动力!!!
  • 获取更多本文的前置知识文章,以及新的有价值的文章,让我们一起成为架构师!
  • 目前已经完成了并发编程、MySQL、spring源码、Mybatis的源码。可以在公众号下方菜单点击查看之前的文章!
  • 这个公众号的所有技术点,会分析的很深入!
  • 这个公众号,无广告!!!

【Spring boot】启动过程源码分析的更多相关文章

  1. [Spring Boot] Spring Boot启动过程源码分析

    关于Spring Boot,已经有很多介绍其如何使用的文章了,本文从源代码(基于Spring-boot 1.5.6)的角度来看看Spring Boot的启动过程到底是怎么样的,为何以往纷繁复杂的配置到 ...

  2. Spring Boot启动过程源码分析--转

    https://blog.csdn.net/dm_vincent/article/details/76735888 关于Spring Boot,已经有很多介绍其如何使用的文章了,本文从源代码(基于Sp ...

  3. Spring启动过程源码分析基本概念

    Spring启动过程源码分析基本概念 本文是通过AnnotationConfigApplicationContext读取配置类来一步一步去了解Spring的启动过程. 在看源码之前,我们要知道某些类的 ...

  4. Spring加载流程源码分析03【refresh】

      前面两篇文章分析了super(this)和setConfigLocations(configLocations)的源代码,本文来分析下refresh的源码, Spring加载流程源码分析01[su ...

  5. spring boot 2.0 源码分析(一)

    在学习spring boot 2.0源码之前,我们先利用spring initializr快速地创建一个基本的简单的示例: 1.先从创建示例中的main函数开始读起: package com.exam ...

  6. Spring Boot 自动配置 源码分析

    Spring Boot 最大的特点(亮点)就是自动配置 AutoConfiguration 下面,先说一下 @EnableAutoConfiguration ,然后再看源代码,到底自动配置是怎么配置的 ...

  7. Android系统默认Home应用程序(Launcher)的启动过程源码分析

    在前面一篇文章中,我们分析了Android系统在启动时安装应用程序的过程,这些应用程序安装好之后,还须要有一个Home应用程序来负责把它们在桌面上展示出来,在Android系统中,这个默认的Home应 ...

  8. Android Content Provider的启动过程源码分析

    本文參考Android应用程序组件Content Provider的启动过程源码分析http://blog.csdn.net/luoshengyang/article/details/6963418和 ...

  9. spring boot 2.0 源码分析(四)

    在上一章的源码分析里,我们知道了spring boot 2.0中的环境是如何区分普通环境和web环境的,以及如何准备运行时环境和应用上下文的,今天我们继续分析一下run函数接下来又做了那些事情.先把r ...

  10. Flume-NG启动过程源码分析(二)(原创)

    在上一节中讲解了——Flume-NG启动过程源码分析(一)(原创)  本节分析配置文件的解析,即PollingPropertiesFileConfigurationProvider.FileWatch ...

随机推荐

  1. Vxe-table 高亮当前行

    需求 1 :设置初始高亮 子组件: 父组件 需求 2 :高亮行的变化,需要把数据传递到兄弟组件中 解决办法:EventBus 参考链接: http://t.csdn.cn/iwOJc main.js ...

  2. reduce累加实现

    与map端的模式类似,map端要重写Mapper方法,reduce端也要重写Reduce方法,这里有一个泛型,我们先看参数类型 分别对应输入keyin,valuein,keyout,valueout. ...

  3. WindTerm:新一代开源免费的终端工具,GitHub星标6.6k+,太酷了!

    继 Tabby.Warp 后,今天再来给大家推荐一款终端神器--WindTerm,完全开源,在 GitHub 上已经收获 6.6k 的 star. https://github.com/kingToo ...

  4. GreatSQL特性介绍及前景展望 | 数据技术嘉年华2021分享PPT发布

    欢迎来到 GreatSQL社区分享的MySQL技术文章,如有疑问或想学习的内容,可以在下方评论区留言,看到后会进行解答 GreatSQL社区原创内容未经授权不得随意使用,转载请联系小编并注明来源. 全 ...

  5. ShadeRec类定义

    这个类主要是用于记录碰撞数据的类,书中已经说的很清楚了.这个类之后会慢慢扩展,会在本随笔中扩展,先定义简单的,方便编译看看效果. 类声明(World是之后主程序中的类,最后测试时再实现): #ifnd ...

  6. C 语言 时间函数使用技巧(汇总)

    time.h 头文件 是 C 语言中 有关 时间的函数所储存的头文件 #include <time.h> 在介绍时间函数用法之前,我们首先要了解在 time.h 头文件中已经声明了的一个结 ...

  7. Word 的页眉、页脚、页码分别是什么?怎么设置?

    页眉:在 Word 文档中,每个页面的顶部区域为页眉.常用于显示文档的附加信息,可以插入时间.图形.公司微标.文档标题.文件名或作者姓名等. 页脚:页脚与页眉的作用相同,都可以作为显示文档的附加信息, ...

  8. Html5新增内容标签

    <canvas>画布</canvas> <audio src=""></audio> <video src="&qu ...

  9. Kotlin快速上手

    一.Kotlin基础 1.数据类型声明 在Kotlin中要定义一个变量需要使用var关键字 //定义了一个可以修改的Int类型变量 var number = 39 如果要定义一个常量可以使用val关键 ...

  10. centos7使用tar包安装mysql5.7

    特别注意: 文档中涉及到密码的都是用的是弱密码,是存在安全风险的,一定要根据自己的情况修改为复杂度更高的密码! centos 7.6 mysql 5.7.31 基础目录: /srv/{app,data ...