初窥spring boot 源码之框架初始化

简介: 从Spring boot 框架初始化深入理解框架是怎么实现的

Spring boot 启动流程

框架初始化

所谓框架初始化,可以说是new 一个  SpringApplication 对象,调用其构造方法时会进行如下步骤的操作!

实际框架初始化流程,

  1. 配置资源加载器,默认启动,资源加载器是null (可以在启动时指定类加载器)
  2. 判断primarySources(servlet项目会将启动主类传入构造方法)不能为空,接下来为SpringApplication 设置primarySources
  3. 调用推断应用所属类型,webflux or none or servlet 应用环境检测 *具体方法在后面有详细写*
  4. 设置系统初始化器-》通过SpringFactoriesLoader 获取ApplicationContextInitializer 相关实现类,主要是扫描jar包下META-INF目录下的spring.factories 文件中的对应类配置
  5. 设置应用监听器-》也是通过SpringFactoriesLoader 获取ApplicationListener 相关实现类,
  6. 配置mainApplicationClass属性,也就是设置main方法所在类

探寻如上流程时,遇到一些有意思的方法

1.SpingApplication 下的 getClassLoader 方法 还会关联到 ClassUtils 中的 getDefaultClassLoader 方法,该方法应该是与双亲委派的代码很是类似。由此看出,我能看出什么呢。这个getDefaultClassLoader 方法呢,就有意思了,如果当前线程有这么一个加载器的话,就用当前线程的,没有则找到ClassUtils类的加载器,如果还是null,那么就使用systemclassloader

2.再一个就是,Throwable 类中会记录一个StackTraceElement,该元素会记录一些内容,比如当前方法所属的类名称。方法名字。。。

这个是在设置main 方法类的时候发现的。

现在深入的研究一波,getSpringFactoriesInstances 方法解析 第4步

1.先获取类加载器

2.通过SpringFactoriesLoader 的loadFactoryNames方法获 调用loadSpringFactories方法 到了set里面,保证名字的唯一

3.createSpringFactoriesInstances 方法进行对象的实例化,先通过ClassUtils.forName 获取到Class对象,再校验是否为需求的子类,通过Class对象及入参创建对应的构造方法对象,通过BeanUtils.instantiatieClass方法创建对象,并把对象放到集合中,返回对象集合。

4.对这些对象,进行一次排序根据其自定义实现的排序逻辑,或者排序器进行排序 很多都是通过order值。

loadSpringFactories流程

  1. 查缓存中是否包含对应数据,如果有直接返回,没有进行下一步
  2. 读取到META-INF目录下的spring.factories 文件
  3. 构造properties 对象所有的kv数据,通过遍历将这些接口及实现的名字拿到,
  4. 之后基于定义的接口名字为key,具体实现为value(value被声明为一个list)(接口的多个实现用“,”分割,有方法进行分割)包装到一个map中,基于这个map 取到type的具体实现类名的集合,将得到的数据放到缓存中
  5. 返回结果

框架启动

框架启动也就是在得到一个springApplication对象以后,进行的run方法的执行。执行会携带启动命令!

探秘框架启动流程

  1. 计时器开始-》new stopwatch().start()
  2. Headless 模式赋值
  3. 获取SpringApplicationRunListeners  spring-boot 框架的sping-factorise 配置文件下指定了默认的启动监听器 并发送一个ApplicationStartingEvent -》 这是个扩展点
  4. 将启动参数包装成一个对象 也就是命令行启动时的参数会通过这一步配置到 applicationArguments 对象中
  5. 配置环境数据,这个方法是配置引入的入口后面进行展开。发送一个 ApplicationEnvironmentPreparedEvent标识环境信息已经准备好了处理忽略的相关内容 -》这也是个有意思的地方
  6. 根据已经加载了配置信息的environment对象创建打印banner对象
  7. 创建应用上下文对象
  8. 创建失败分析器
  9. prepareContext 准备上下文对象 下面有详解
  10. refreshContext(context); 对上下文进行刷新,bean的初始化都在这个方法中,后面会拓展。下一个方法在默认启动中没有具体实现代码逻辑,可能是为了实现设置的。
  11. 停止计时-》 打印启动哪个类 用了多长时间 的日志
  12. 发送started事件
  13. 调用runners方法 对启动加载器进行处理
  14. 如果以上从第4步到12步有异常或者错误发生,就会通过失败分析器,发出相关失败信息。
  15. 没啥事儿再发个框架正常启动的事件,然后将上下文对象

image.png

以上为框架初始化及启动相关流程,大家加油。

相关文章
|
20小时前
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
|
20小时前
|
XML Java 数据格式
Spring底层架构源码解析(二)
Spring底层架构源码解析(二)
|
1天前
|
开发框架 Java API
「SpringBrick快速入门指南」:一款基于Spring Boot的高级插件化开发框架
「SpringBrick快速入门指南」:一款基于Spring Boot的高级插件化开发框架
5 0
|
1天前
|
人工智能 Java API
阿里云开源 AI 应用开发框架:Spring AI Alibaba
阿里云开源 Spring AI Alibaba,旨在帮助 Java 开发者快速构建 AI 应用,共同构建物理新世界。
|
1天前
|
XML Java 数据格式
手动开发-简单的Spring基于注解配置的程序--源码解析
手动开发-简单的Spring基于注解配置的程序--源码解析
4 0
|
1天前
|
XML Java 数据格式
手动开发-简单的Spring基于XML配置的程序--源码解析
手动开发-简单的Spring基于XML配置的程序--源码解析
3 0
|
2天前
|
监控 Java 开发者
BeanPostProcessor:Spring框架的灵活扩展机制
【10月更文挑战第4天】在Spring框架中,BeanPostProcessor接口是一个非常重要的扩展点,它允许开发者在Spring容器实例化、依赖注入以及初始化Bean的过程中插入自定义逻辑。
8 0
|
11月前
|
Java Spring
spring框架之AOP模块(面向切面),附带通知类型---超详细介绍
spring框架之AOP模块(面向切面),附带通知类型---超详细介绍
109 0
|
11月前
|
缓存 监控 Java
Spring框架之AOP(面向切面编程)
Spring框架之AOP(面向切面编程)
51 0
|
3月前
|
分布式计算 Java MaxCompute
详解 Java 限流接口实现问题之在Spring框架中使用AOP来实现基于注解的限流问题如何解决
详解 Java 限流接口实现问题之在Spring框架中使用AOP来实现基于注解的限流问题如何解决