欢迎关注头条号:java小马哥

周一至周日早九点半!下午三点半!精品技术文章准时送上!!!

精品学习资料获取通道,参见文末

用了差不多两年的SpringBoot了,可以说对SpringBoot已经很熟了,但是仔细一想SpringBoot的启动流程,还是让自己有点懵逼,不得不说是自己工作和学习的失误,所以以此文对SpringBoot的启动流程略作记录。

此文的SpringBoot启动流程分析是基于SpringBoot 1.x的,SpringBoot 2.x的启动流程与1.x的略有不同,后续再进行补充分析。

核心注解@SpringBootApplication

每个SpringBoot应用,都有一个入口类,标注@SpringBootApplication注解。

1

2

3

4

5

6

@SpringBootApplication

public class DemoApplication {

public static void main(String[] args) {

SpringApplication.run(DemoApplication.class, args);

}

}

点开@SpringBootApplication的源码,可以看到这个注解其实包含了@SpringBootConfiguration,@EnableAutoConfiguration,@ComponentScan三个注解。

下面对这三个注解简单解释解释。

  • @SpringBootConfiguration
  • 对于这个注解不做解释,将这个注解点进去,发现还有@Configuration注解,对于@Configuration注解,用过Spring或SpringBoot的基本上都不陌生,标注了@Configuration的类相当于Spring中的配置XML,不过SpringBoot社区推荐使用JavaConfig,所以@Configuration就构建出了一个基础JavaConfig的Ioc容器。
  • @EnableAutoConfiguration
  • Spring中有很多Enable*的注解,表示开启某项东西,如@EnableSchuduling。所以看这个注解的名字就知道是开启自动配置。这是一个复合注解,其中最主要的还是@Import,借助于EnableAutoConfigurationImportSelector,将所有符合自动配置条件的Bean加载到Ioc容器里。
  • SpringBoot加载自动配置的方式有两种(目前我知道的):
  • 在classpath下新建META-INF/spring.factories文件,将标注了@Configuration的类的全路径配置到此文件中,如:

1

2

3

org.springframework.boot.autoconfigure.EnableAutoConfiguration=

com.quartz.config.QuartzBeanConfiguration,

com.quartz.config.QuartzAutoConfiguration

  • 在启动时,通过SpringFactoriesLoader工具类,将所有META-INF目录下的spring.factories文件中的配置类加载到Ioc容器里。
  • 使用@Import,将配置类加载到Ioc容器里。

1

2

3

4

5

6

@Target(ElementType.TYPE)

@Retention(RetentionPolicy.RUNTIME)

@Documented

@Import({QuartzAutoConfiguration.class})

public @interface EnableQuartz {

}

  • 使用@Import导入的类必须满足以下任意一个要求:
  1. 导入的类使用@Configuration进行标注
  2. 导入的类中至少有一个使用@Bean标准的方法
  3. 导入的类实现了ImportSelector接口
  4. 导入的类实现了ImportBeanDefinitionRegistrar接口
  • @ComponentScan
  • 看到这个注解,可以回想一下以前使用SpringMVC时,xml配置文件里的一个标签

1

  • 不过这个注解一般不需要手动指定扫描的包路径,它默认会从标注了@ComponentScan的类所在包往下查找,将标注了如@Component,@Service等Bean加载到Ioc容器里。

自动配置核心类SpringFactoriesLoader

上面在说@EnableAutoConfiguration的时候有说META-INF下的spring.factories文件,那么这个文件是怎么被spring加载到的呢,其实就是SpringFactoriesLoader类。

SpringFactoriesLoader是一个供Spring内部使用的通用工厂装载器,SpringFactoriesLoader里有两个方法,

1

2

3

4

// 加载工厂类并实例化

public static List loadFactories(Class factoryClass, ClassLoader classLoader) {}

// 加载工厂类的类名

public static List loadFactoryNames(Class> factoryClass, ClassLoader classLoader) {}

在这个SpringBoot应用启动过程中,SpringFactoriesLoader做了以下几件事:

  1. 加载所有META-INF/spring.factories中的Initializer
  2. 加载所有META-INF/spring.factories中的Listener
  3. 加载EnvironmentPostProcessor(允许在Spring应用构建之前定制环境配置)
  4. 接下来加载Properties和YAML的PropertySourceLoader(针对SpringBoot的两种配置文件的加载器)
  5. 各种异常情况的FailureAnalyzer(异常解释器)
  6. 加载SpringBoot内部实现的各种AutoConfiguration
  7. 模板引擎TemplateAvailabilityProvider(如Freemarker、Thymeleaf、Jsp、Velocity等)

总得来说,SpringFactoriesLoader和@EnableAutoConfiguration配合起来,整体功能就是查找spring.factories文件,加载自动配置类。

整体启动流程

在我们执行入口类的main方法之后,运行SpringApplication.run,后面new了一个SpringApplication对象,然后执行它的run方法。

1

2

3

public static ConfigurableApplicationContext run(Object[] sources, String[] args) {

return new SpringApplication(sources).run(args);

}

初始化SpringApplication类

创建一个SpringApplication对象时,会调用它自己的initialize方法

1

2

3

4

5

6

7

8

9

10

11

12

13

14

private void initialize(Object[] sources) {

if (sources != null && sources.length > 0) {

this.sources.addAll(Arrays.asList(sources));

}

// 根据标志类javax.servlet.Servlet,org.springframework.web.context.ConfigurableWebApplicationContext是否存在,判断是否是web环境

this.webEnvironment = deduceWebEnvironment();

// 通过SpringFactoriesLoader,获取到所有META-INF/spring.factories中的ApplicationContextInitializer,并实例化

setInitializers((Collection) getSpringFactoriesInstances(

ApplicationContextInitializer.class));

// 通过SpringFactoriesLoader,获取到所有META-INF/spring.factories中的ApplicationListener,并实例化

setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));

// 获取执行当前main方法的类,也就是启动类

this.mainApplicationClass = deduceMainApplicationClass();

}

注 : 各方法内部执行逻辑就不做说明了,比较简单,需要的读者可自行点进源码查看

  1. 根据classpath里是否存在某个特征类(javax.servlet.Servlet,org.springframework.web.context.ConfigurableWebApplicationContext)来判断是否需要创建一个为Web应用使用的ApplicationContext。
  2. 使用SpringFactoriesLoader在应用的classpath下的所有META-INF/spring.factories中查找并加载所有可用的ApplicationContextInitializer。
  3. 使用SpringFactoriesLoader在应用的classpath下的所有META-INF/spring.factories中查找并加载所有可用的ApplicationListener。
  4. 设置main方法的定义类

执行核心run方法

初始化initialize方法执行完之后,会调用run方法,开始启动SpringBoot。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

public ConfigurableApplicationContext run(String... args) {

// 启动任务执行的时间监听器

StopWatch stopWatch = new StopWatch();

stopWatch.start();

ConfigurableApplicationContext context = null;

FailureAnalyzers analyzers = null;

// 设置系统java.awt.headless属性,确定是否开启headless模式(默认开启headless模式)

configureHeadlessProperty();

// 通过SpringFactoriesLoader,获取到所有META-INF/spring.factories下的SpringApplicationRunListeners并实例化

SpringApplicationRunListeners listeners = getRunListeners(args);

// 开始广播启动

listeners.started();

try {

// 创建SpringBoot默认启动参数对象

ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);

// 根据启动参数创建并配置Environment(所有有效的配置,如Profile),并遍历所有的listeners,广播启动环境已准备

ConfigurableEnvironment environment = prepareEnvironment(listeners,applicationArguments);

// 打印启动图案

Banner printedBanner = printBanner(environment);

// 根据标志类(上面有提到过),创建对应类型的ApplicationContext

context = createApplicationContext();

// 创建异常解析器(当启动失败时,由此解析器处理失败结果)

analyzers = new FailureAnalyzers(context);

// 准备Spring上下文环境

// 在这个方法中,主要完成了以下几件事:

// 1、设置SpringBoot的环境配置(Environment)

// 2、注册Spring Bean名称的序列化器BeanNameGenerator,并设置资源加载器ResourceLoader

//3、加载ApplicationContextInitializer初始化器,并进行初始化

//4、统一将上面的Environment、BeanNameGenerator、ResourceLoader使用默认的Bean注册器进行注册

prepareContext(context, environment, listeners, applicationArguments,printedBanner);

// 注册一个关闭Spring容器的钩子

refreshContext(context);

// 获取当前所有ApplicationRunner和CommandLineRunner接口的实现类,执行其run方法

// ApplicationRunner和CommandLineRunner功能基本一样,在Spring容器启动完成时执行,唯一不同的是ApplicationRunner的run方法入参是ApplicationArguments,而CommandLineRunner是String数组

afterRefresh(context, applicationArguments);

// 通知所有listener,Spring容器启动完成

listeners.finished(context, null);

// 停止时间监听器

stopWatch.stop();

if (this.logStartupInfo) {

new StartupInfoLogger(this.mainApplicationClass)

.logStarted(getApplicationLog(), stopWatch);

}

return context;

} catch (Throwable ex) {

// 启动有异常时,调用异常解析器解析异常信息,根据异常级别,判断是否退出Spring容器

handleRunFailure(context, listeners, analyzers, ex);

throw new IllegalStateException(ex);

}

}

  1. 首先遍历执行所有通过SpringFactoriesLoader,在当前classpath下的META-INF/spring.factories中查找所有可用的SpringApplicationRunListeners并实例化。调用它们的started()方法,通知这些监听器SpringBoot应用启动。
  2. 创建并配置当前SpringBoot应用将要使用的Environment,包括当前有效的PropertySource以及Profile。
  3. 遍历调用所有的SpringApplicationRunListeners的environmentPrepared()的方法,通知这些监听器SpringBoot应用的Environment已经完成初始化。
  4. 打印SpringBoot应用的banner,SpringApplication的showBanner属性为true时,如果classpath下存在banner.txt文件,则打印其内容,否则打印默认banner。
  5. 根据启动时设置的applicationContextClass和在initialize方法设置的webEnvironment,创建对应的applicationContext。
  6. 创建异常解析器,用在启动中发生异常的时候进行异常处理(包括记录日志、释放资源等)。
  7. 设置SpringBoot的Environment,注册Spring Bean名称的序列化器BeanNameGenerator,并设置资源加载器ResourceLoader,通过SpringFactoriesLoader加载ApplicationContextInitializer初始化器,调用initialize方法,对创建的ApplicationContext进一步初始化。
  8. 调用所有的SpringApplicationRunListeners的contextPrepared方法,通知这些Listener当前ApplicationContext已经创建完毕。
  9. 最核心的一步,将之前通过@EnableAutoConfiguration获取的所有配置以及其他形式的IoC容器配置加载到已经准备完毕的ApplicationContext。
  10. 调用所有的SpringApplicationRunListener的contextLoaded方法,加载准备完毕的ApplicationContext。
  11. 调用refreshContext,注册一个关闭Spring容器的钩子ShutdownHook,当程序在停止的时候释放资源(包括:销毁Bean,关闭SpringBean的创建工厂等)
  12. 注: 钩子可以在以下几种场景中被调用:
  13. 1)程序正常退出
  14. 2)使用System.exit()
  15. 3)终端使用Ctrl+C触发的中断
  16. 4)系统关闭
  17. 5)使用Kill pid命令杀死进程
  18. 获取当前所有ApplicationRunner和CommandLineRunner接口的实现类,执行其run方法
  19. 遍历所有的SpringApplicationRunListener的finished()方法,完成SpringBoot的启动。

封面图源网络,侵权删除)

私信头条号,发送:“资料”,获取更多“秘制” 精品学习资料

如有收获,请帮忙转发,您的鼓励是作者最大的动力,谢谢!

一大波微服务、分布式、高并发、高可用的原创系列文章正在路上,

欢迎关注头条号:java小马哥

周一至周日早九点半!下午三点半!精品技术文章准时送上!!!

十余年BAT架构经验倾囊相授

springboot 自动装配_Java互联网架构-SpringBoot自动装配核心源码剖析相关推荐

  1. SpringBoot初始化过程核心源码剖析

    前言 首先我们启动一个SpringBoot项目是怎么启动的?是依着Main方法中的SpringApplication.run(args);这个方法来进行启动的,很多人都对这个有点误解,以为是@Spri ...

  2. java获取当前周一_Java互联网架构-Spring IOC源码分析

    欢迎关注头条号:java小马哥 周一至周日下午三点半!精品技术文章准时送上!!! 精品学习资料获取通道,参见文末 源码介绍之前,看几个问题: Bean的承载对象是什么? Bean的定义如何存储的? B ...

  3. Java互联网架构 百度云_java互联网架构师

    资源内容: java互联网架构师|____014_互联网架构视频第二期(017).rar|____013_互联网架构视频第二期(016).rar|____012_互联网架构视频第二期(015).rar ...

  4. 一箭双雕 刷完阿里P8架构师spring学习笔记+源码剖析,涨薪8K

    关于Spring的叙述: 我之前死磕spring的时候,刷各种资料看的我是一头雾水的,后面从阿里的P8架构师那里拿到这两份资料,从源码到案例详细的讲述了spring的各个细节,是我学Spring的启蒙 ...

  5. 【Netty源码解析】Netty核心源码和高并发、高性能架构设计精髓

    Netty线程模型图 Netty线程模型源码剖析图 图链接:https://www.processon.com/view/link/5dee0943e4b079080a26c2ac Netty高并发高 ...

  6. aop springboot 传入参数_java相关:springboot配置aop切面日志打印过程解析

    java相关:springboot配置aop切面日志打印过程解析 发布于 2020-3-31| 复制链接 摘记: 这篇文章主要介绍了springboot配置aop切面日志打印过程解析,文中通过示例代码 ...

  7. hashmap 存的是对象的引用地址_Java互联网架构-面试虐我千百遍HashMap源码真讨厌...

    在java的容器集合中,hashmap的使用频率可以说是相当高的.不过对于hashmap的存(put())以及取(get())的原理可能很多人还不大清楚,今天,我就给大家介绍下它是如何存如何取的. # ...

  8. springboot 技术图谱_java后台(Springboot)开发知识图谱高频技术汇总-学习路线...

    [原创]java后台(Springboot)开发知识图谱&&高频技术汇总 1.引言: 学习一个新的技术时,其实不在于跟着某个教程敲出了几行.几百行代码,这样你最多只能知其然而不知其所以 ...

  9. 小马源码_Java互联网架构-重新认识Java8-HashMap-不一样的源码解读

    欢迎关注头条号:java小马哥 周一至周日早九点半!下午三点半!精品技术文章准时送上!!! 精品学习资料获取通道,参见文末 看源码前我们必须先知道一下ConcurrentHashMap的基本结构.Co ...

最新文章

  1. 电脑桌面便签小工具_电脑桌面工作任务提醒软件有哪些?多端同步提醒办公软件试试云便签...
  2. 大型网站技术架构(2):架构要素和高性能架构
  3. rabbitmq之partitions
  4. [工作积累] shadow map问题汇总
  5. 排球积分程序(三)——模型类的设计
  6. ubuntu之sudo apt-get update提示Could not connect to 127.0.0.1:8081 (127.0.0.1)解决办法
  7. “新基建”提速,工业互联网大数据发展迎新机遇
  8. 职业生涯第一次:老板让我写个 BUG!
  9. 零基础带你学习MySQL—多表查询笛卡尔集(二十)
  10. 导入一个maven项目出现红色叉号的解决办法
  11. javascript对数组的操作
  12. 查询输出优秀人数_sql 第五关多表查询
  13. vlfeat各种版本下载链接:
  14. 视频截帧 php,php截取视频指定帧为图片_PHP
  15. LibEvent中文帮助文档--第1、2、3、4章
  16. LintCode 38: Search a 2D Matrix II
  17. mac系统 查找英文目录
  18. Redis主从, 哨兵, Lettuce(二)
  19. DOS命令:comp
  20. 语义分割-地表建筑物识别的一种解决方案

热门文章

  1. 迁移学习(Transfer learning)、重用预训练图层、预训练模型库
  2. 更高效的PacBio长read纠错算法的研究
  3. 生物信息学常见数据格式 • fasta • fastq • gff/gtf 练习题
  4. php doss_php下ddos攻击与防范代码
  5. java accept encoding_Accept-Encoding gzip 乱码 和Okhttp的解决方法
  6. js折线图设置y轴刻度_手绘风格的 JS 图表库:Chart.xkcd
  7. windows10 删除文件 的权限才能对此文件夹进行更改 解决办法
  8. 【矩阵运算c++实现】矩阵封装实现Matrix类
  9. LeetCode 175. Combine Two Tables--Database--数据库题目
  10. java判断两个int相等_Java 判断两个变量是否相等