SpringBoot源码分析系列之五:再探自动装配原理

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 我们从程序运行的角度把自动配置的主要的源码进行了分析,过滤了一些步骤,主要是调用层级太多容易让人抓不住重点,这边也是给大家看源码的一点小技巧就是抓大放小,抓住主要流程,忽略掉小的细节部分,否则我们就会陷入到源码的汪洋之中无法自拔。

引言

自动装配原理是SpringBoot的一大特性,网上也有各种分析文章,但是大同小异,总感觉没有说到点子上,或者说只是说了一半,并没有就完整的流程进行详细说明。

  • 什么是自动装配
  • 自动装配源码分析
  • 总结


一、什么是自动装配

SpringBoot诞生之前,我们利用Spring进行应用开发的时候,研发同学需要花费大量精力去定义各类模板化的配置文件,十分繁琐以及机械。Spring使用Bean Factory以及动态代理实现各模块之间的解耦,它通过配置文件将需要加载的bean扫描到Spring容器中。而SpringBoot正是将这种xml解析配置的过程,通过注解自动配置的方式来进行替换,它根据定义在classpath下的类以及jar包中的META-INF目录下的spring.factories定义的完全限定名的类,自动生成对应的bean,同时将其加载到Spring的ApplicationContext中。

二、自动装配源码分析

网上关于SpringBoot自动装配原理的各种分析文章也非常多,它们大部分的分析思路主要从启动类上的@SpringBootApplication注解开始分析,通过它找到@EnableAutoConfiguration注解,再找到AutoConfigurationImportSelector,最后找到以下这段代码。通过加载META-INF目录下的spring.factories来加载对应的配置类。

 protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
        List<String> configurations = SpringFactoriesLoader.loadFactoryNames(this.getSpringFactoriesLoaderFactoryClass(), this.getBeanClassLoader());
        Assert.notEmpty(configurations, "No auto configuration classes found in META-INF/spring.factories. If you are using a custom packaging, make sure that file is correct.");
        return configurations;
    }

这个过程仿佛对自动装配原理进行了分析说明。但是实际上这个流程完全是从人的角度去理解的,而不是从程序运行的角度去分析的。另外SpringBoot到底是怎么通过主类解析对应注解的,这些重要的解析过程也没有进行分析。因此这次我们换个角度,从程序运行的角度去探究自动装配的过程。


一切的故事都还是要从启动类开始说起,但是我们这次先不拿@SpringBootApplication说事。我们从run方法进行入手分析。如下图所示:

image.png

通过上篇文章可知,运行run方法之前,首先通过构造函数构建了SpringApplication实例。在构造函数中传入了primarySources参数,即为当前的运行主类。

1、primarySources

image.png

2、设置主应用类

此方法首先获取运行过程中的堆栈信息,如下图所示,其中包含了当前运行的方法。但是它需要的的方法名称是main方法。通过找到main方法,获取到启动主类FrameApplication,同时对启动主类进行加载。这样在实例化的时候就明确了当前的启动主类。

image.png

3、将应用主类加载到ApplicationContext

在启动run方法中做了很多事情,包括设置属性、获取监听器,启动监听器等。这里主要关注自动配置的加载过程,因此这些过程不再进行赘述。prepareContext时候进行类的加载。根据不同的类型执行不同的加载流程。

image.png

由于当前的Source是来自主类的FrameApplication,因此执行的是第一个分支进行加载。在加载方法中判断了当前的类是不是组件,如果是的话则进行注册。

image.png

判断组件的方法就是判断当前类的组合注解中是否包含@Component的注解。

image.png

也就是说它会根据主类的注解,一层一层往上去寻找,检查是否包含@Component的注解。启动主类的注解包含关系可以参见下图所示:

image.png

通过下图可知,在prepareContext阶段,已经将运行主类注册到了Spring容器当中。

image.png

本文主要分析自动装配解析的过程,其中涉及到的Spring容器启动等的内容不再详细阐述了。因此下图给出了调用栈的信息,不再一步一步debug代码。

image.png

4、解析自动配置

ConfigurationClassParser中对@Import注解进行解析,获取到了 AutoConfigurationImportSelector.class以及Registrar.class这两个类。

image.png

AutoConfigurationImportSelector 类中,通过获取META-INF目录下的spring.factories 加载所有的124个自动配置类,当然在后续又对重复的以及不需要的自动配置类进行了去除,只保留了启动运行需要的主要自动配置类。

image.png

至此,SpringBoot启动时完成的自动配置流程算是讲清楚了。

三、总结

通过上文的分析,我们从程序运行的角度把自动配置的主要的源码进行了分析,过滤了一些步骤,主要是调用层级太多容易让人抓不住重点,这边也是给大家看源码的一点小技巧就是抓大放小,抓住主要流程,忽略掉小的细节部分,否则我们就会陷入到源码的汪洋之中无法自拔。

我们再通过一张流程图来回顾下整个过程:

image.png

相关文章
|
24天前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
1月前
|
前端开发 Java
表白墙/留言墙 —— 初级SpringBoot项目,练手项目前后端开发(带完整源码) 全方位全步骤手把手教学
文章通过一个表白墙/留言墙的初级SpringBoot项目实例,详细讲解了如何进行前后端开发,包括定义前后端交互接口、创建SpringBoot项目、编写前端页面、后端代码逻辑及实体类封装的全过程。
75 3
表白墙/留言墙 —— 初级SpringBoot项目,练手项目前后端开发(带完整源码) 全方位全步骤手把手教学
|
1月前
|
前端开发 Java 数据安全/隐私保护
用户登录前后端开发(一个简单完整的小项目)——SpringBoot与session验证(带前后端源码)全方位全流程超详细教程
文章通过一个简单的SpringBoot项目,详细介绍了前后端如何实现用户登录功能,包括前端登录页面的创建、后端登录逻辑的处理、使用session验证用户身份以及获取已登录用户信息的方法。
189 2
用户登录前后端开发(一个简单完整的小项目)——SpringBoot与session验证(带前后端源码)全方位全流程超详细教程
|
8天前
|
XML Java 开发者
Spring Boot开箱即用可插拔实现过程演练与原理剖析
【11月更文挑战第20天】Spring Boot是一个基于Spring框架的项目,其设计目的是简化Spring应用的初始搭建以及开发过程。Spring Boot通过提供约定优于配置的理念,减少了大量的XML配置和手动设置,使得开发者能够更专注于业务逻辑的实现。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,为开发者提供一个全面的理解。
19 0
|
12天前
|
Java Spring
SpringBoot自动装配的原理
在Spring Boot项目中,启动引导类通常使用`@SpringBootApplication`注解。该注解集成了`@SpringBootConfiguration`、`@ComponentScan`和`@EnableAutoConfiguration`三个注解,分别用于标记配置类、开启组件扫描和启用自动配置。
49 17
|
23天前
|
JavaScript Java 项目管理
Java毕设学习 基于SpringBoot + Vue 的医院管理系统 持续给大家寻找Java毕设学习项目(附源码)
基于SpringBoot + Vue的医院管理系统,涵盖医院、患者、挂号、药物、检查、病床、排班管理和数据分析等功能。开发工具为IDEA和HBuilder X,环境需配置jdk8、Node.js14、MySQL8。文末提供源码下载链接。
|
1月前
|
缓存 Java Spring
servlet和SpringBoot两种方式分别获取Cookie和Session方式比较(带源码) —— 图文并茂 两种方式获取Header
文章比较了在Servlet和Spring Boot中获取Cookie、Session和Header的方法,并提供了相应的代码实例,展示了两种方式在实际应用中的异同。
160 3
servlet和SpringBoot两种方式分别获取Cookie和Session方式比较(带源码) —— 图文并茂 两种方式获取Header
|
1月前
|
Java Spring 容器
springboot @RequiredArgsConstructor @Lazy解决循环依赖的原理
【10月更文挑战第15天】在Spring Boot应用中,循环依赖是一个常见问题,当两个或多个Bean相互依赖时,会导致Spring容器陷入死循环。本文通过比较@RequiredArgsConstructor和@Lazy注解,探讨它们解决循环依赖的原理和优缺点。@RequiredArgsConstructor通过构造函数注入依赖,使代码更简洁;@Lazy则通过延迟Bean的初始化,打破创建顺序依赖。两者各有优势,需根据具体场景选择合适的方法。
57 4
|
1月前
|
架构师 Java 开发者
得物面试:Springboot自动装配机制是什么?如何控制一个bean 是否加载,使用什么注解?
在40岁老架构师尼恩的读者交流群中,近期多位读者成功获得了知名互联网企业的面试机会,如得物、阿里、滴滴等。然而,面对“Spring Boot自动装配机制”等核心面试题,部分读者因准备不足而未能顺利通过。为此,尼恩团队将系统化梳理和总结这一主题,帮助大家全面提升技术水平,让面试官“爱到不能自已”。
得物面试:Springboot自动装配机制是什么?如何控制一个bean 是否加载,使用什么注解?
|
2月前
|
Java 应用服务中间件 API
Vertx高并发理论原理以及对比SpringBoot
Vertx 是一个基于 Netty 的响应式工具包,不同于传统框架如 Spring,它的侵入性较小,甚至可在 Spring Boot 中使用。响应式编程(Reactive Programming)基于事件模式,通过事件流触发任务执行,其核心在于事件流 Stream。相比多线程异步,响应式编程能以更少线程完成更多任务,减少内存消耗与上下文切换开销,提高 CPU 利用率。Vertx 适用于高并发系统,如 IM 系统、高性能中间件及需要较少服务器支持大规模 WEB 应用的场景。随着 JDK 21 引入协程,未来 Tomcat 也将优化支持更高并发,降低响应式框架的必要性。
Vertx高并发理论原理以及对比SpringBoot

热门文章

最新文章