SpringBoot自动装配原理(附面试快速答法)

文章目录

  • SpringBoot自动装配原理
    • 1. 从调用SpringApplication构造器方法开始
    • 2. 解析启动类
    • 4.按需装配
      • 4.1 分析dubbo自动装配
    • 5. 如果定义自己的starter
    • 6. 面试答法

SpringBoot自动装配原理

之前面试被问到这个题目,只会答一些spi、@AutoConfigration注解、@Import之类的,感觉面试官并不是很满意,自己也还停留在八股文的水平,最近有时间了,仔细总结一下

1. 从调用SpringApplication构造器方法开始

以下源码分析基于springboot2.6.x

首先一切的开始都是从这个方法开始的SpringApplication.run(),所以说自动装箱的核心就是这个run方法的执行过程

首先我们应该带着问题看这个方法的执行

  • 是否需要创建IOC容器,需要创建那些Bean
  • 创建Bean之前的准备工作

最好的看源码的方式是通过debug的方式,我们在SpringApplication.run()上面打一个断点,然后一步一步的分析

image-20221111230725572

首先第一步,SpringApplication.run()方法需要传入两个参数,第一个参数就是启动类本身,用于在之后解析启动类(解析标记的注解、启动类作为一个配置类,也需要解析),后面那个args是传入的虚拟机参数

再往下一层能够看到new SpringApplication(primarySources).run(args),当我们看到new关键字的时候我们处了知道创建了一个对象之外,还应该注意到他调用了SpringApplication这个类的构造器,构造器一般会用来加载一些配置

image-20221111235509417

启动类里面的这一行其实在表示当前Spring要用什么web方式,在JavaWeb开发中一般常用的就是Servlet程序,其实就对应着SpringMVC,Spring还提供了SpringWebflux来进行响应式响应式编程

image-20221112000300822

/**
 * The application should not run as a web application and should not start an
 * embedded web server.
 */
NONE,
/**
 * The application should run as a servlet-based web application and should start an
 * embedded servlet web server.
 */
SERVLET,
/**
 * The application should run as a reactive web application and should start an
 * embedded reactive web server.
 */
REACTIVE;

接下来就是非常重要的两行代码了,这里就涉及到了spring.factories文件到底是怎么加载的问题了

image-20221112000816135

一看名字就能看出来

  • 设置初始化器
  • 设置监听器

这里面的getSpringFactoriesInstances就是去加载spring.factories带给我们自动装箱的bean实例的

image-20221112001616730

通过org.springframework.context.ApplicationContextInitializer来加载自动装箱文件

image-20221112001827828

再然后获取spring.factories文件的资源路径

image-20221112002246735

其实这个资源文件FACTORIES_RESOURCE_LOCATION也早已经定义好了

image-20221112002326264

然后就到了加载配置文件里面配置好的bean了

image-20221112002847601

接下来就是通过反射创建实例并返回的操作

到目前为止,其实还没有牵涉到自动装配的东西

2. 解析启动类

我们继续debug,直接到解析启动类的地方

org.springframework.boot.SpringApplication#prepareContext

看这里,就是去加载启动类

image-20221112005418594

一路进入到这里

org.springframework.boot.BeanDefinitionLoader#load()

image-20221112005642511

开始加载我们的启动类,可以看到其实我们的启动类可以定义不止一个,进入load方法

image-20221112010055876

这里看到一个方法isEligible(source),即有资格的的bean,这个方法在低版本的boot中是isComponent(source),很好理解,即这个bean必须要被@Component标记才能被注册

准备工作完成之后接下来进入到我们非常重要的一个方法refreshContext(context)方法

image-20221112102518397

接下来会落到这个方法里面

image-20221112102749762

当我们在Spring源码中看到refresh方法的时候,其实我们应该知道这肯定跟SpringIOC容器有关了,我们在org.springframework.context.support.AbstractApplicationContext#refresh中其实能够看到13个方法,在Spring中所有带有refresh方法最终的实现基本上都是在这里

image-20221112103044669

我们看到其中的一个方法postProcessBeanFactory(beanFactory)

这个方法就是做增强的,在Spring中看到postProcessBean开头的方法,我们一般喜欢将其称之为后置增强器

我们来到这里,其中configCandidates用来存放启动类

org.springframework.context.annotation.ConfigurationClassPostProcessor#processConfigBeanDefinitions

在这里会去找到所有匹配的能够被解析的类,可以看到已经拿到我们的启动类了

image-20221113155249642

image-20221113155444972

image-20221113155648133

继续往下,我们就能看到解析启动类的地方

-> org.springframework.context.annotation.ConfigurationClassParser#processConfigurationClass
-> org.springframework.context.annotation.ConfigurationClassParser#doProcessConfigurationClass

image-20221113160220051

我们来看一下解析@Import注解的地方,因为在实际开发中自动装箱我们用这个注解比较多

    org.springframework.context.annotation.ConfigurationClassParser#collectImports

在这里的方法中递归解析

image-20221113160624860

拿到import注解后我们在来到这里org.springframework.boot.autoconfigure.AutoConfigurationImportSelector.AutoConfigurationGroup#process

image-20221113162737875

然后再到这里面就会去加载spring.factories文件了

SpringBoot 定义了一套接口规范,这套规范规定:SpringBoot 在启动时会扫描外部引用 jar 包中的META-INF/spring.factories文件,将文件中配置的类型信息加载到 Spring 容器

org.springframework.core.io.support.SpringFactoriesLoader#loadSpringFactories

比较精彩的一点是他会去拿一下之前已经加载过的缓存,避免重复加载

image-20221113163547837

当然也不会全部加载进去,如果项目中没有依赖,会将这些bean排除掉

image-20221113163824676

那么到底哪些bean会被加载?哪些bean会被过滤掉呢?我们继续往下看

4.按需装配

那么面试官可能会问:怎么排除META-INF/spring.factories里不需要的bean呢?

我们可以在springboot官网上找到答案

image-20230411105606019

我们的自动装配依赖条件注解,来判断哪些bean需要加载进入IOC容器,这一类条件注解一般是以@Conditional开头

image-20230411110921326

4.1 分析dubbo自动装配

我们来看一下dubbo所依赖的bean是如何自动装配到ioc容器中的

首先我们需要引入dubbo的starter

<!-- dubbo -->
<dependency>
    <groupId>org.apache.dubbo</groupId>
    <artifactId>dubbo-spring-boot-starter</artifactId>
    <version>3.0.7</version>
</dependency>

<dependency>
    <groupId>org.apache.dubbo</groupId>
    <artifactId>dubbo-dependencies-zookeeper-curator5</artifactId>
    <version>${dubbo.version}</version>
    <type>pom</type>
</dependency>

既然是一个starter,肯定得准守springboot约定大于配置的约定,在对应的jar包下一定有个**META-INF/spring.factories**文件

image-20230411115507105

org.springframework.core.io.support.SpringFactoriesLoader#loadSpringFactories方法加载完毕后,这些配置的bean就会通过反射进行加载

我们分析一下这个bean,点进去进行查看

image-20230411112706794

image-20230411113943009

但是当我们去查看最终注入到ioc容器里的bean的时候,却没有发现这些bean,甚至一个关于dubbo的bean都没有

image-20230411113853748

这是因为我们的项目中并没有添加对dubbo的条件注解,没有按需进入注入

这时候其实我们需要的是添加条件注解

image-20230411115847128

image-20230411115915590

image-20230411115937449

可以看到,当项目中有条件注解时,才能加入到ioc容器

5. 如果定义自己的starter

定义的步骤在springboot官网上非常详细

  • Createing Your Own Auto-configuration

6. 面试答法

首先自动装配中最重要的三个类,回答的时候要沿着这三个方法去回答

  • BFPP: BeanFactoryPostProcessor
  • BPP: BeanPostProcessor
  • BDRPP:BeanDefinitionRegistryPostProcessor

第一步:自动装配是什么?解决了那些问题

第二步:自动装配的过程

  1. 当启动springboot应用程序的时候,会先创建SpringApplication的对象,在对象的构造方法中会进行某些参数的初始化工作,最主要的是判断当前应用程序的类型以及初始化器和监听器,在这个过程中会加载整个应用程序中的spring.factories文件,将文件的内容放到缓存对象中,方便后续获取。
  2. SpringApplication对象创建完成之后,开始执行run方法,来完成整个启动,启动过程中最主要的有两个方法,第一个叫做prepareContext,第二个叫做refreshContext,在这两个关键步骤中完整了自动装配的核心功能,前面的处理逻辑包含了上下文对象的创建,banner的打印,异常报告期的准备等各个准备工作,方便后续来进行调用。
  3. 在prepareContext方法中主要完成的是对上下文对象的初始化操作,包括了属性值的设置,比如环境对象,在整个过程中有一个非常重要的方法,叫做load,load主要完战一件事,将当前启动类做为一个beanDefinition注册到registry中,方便后续在进行BeanFactoryPostProcessor调用执行的时候,找到对应的主类,来完成@SpringBootApplicaiton@EnableAutoConfiguration等注解的解析工作
  4. 在refreshContext方法中会进行整个容器刷新过程,会调用中spring中的refresh方法,refresh中有13个非常关键的方法,来完成整个spring应用程序的启动,在自动装配过程中,会调用invokeBeanFactoryPostProcessor方法,在此方法中主要是对ConfigurationClassPostProcessor类的处理,这次是BFPP的子类也是BDRPP的子类,在调用的时候会先调用BDRPP中的postProcessBeanDefinitionRegistry方法,然后调用postProcessBeanFactory方法,在执行postProcesskeanDefinitionRegistry的时候回解析处理各种注解,包含@PropertySource,@ComponentScan,@ComponentScans,@Bean,@lmport等注解,最主要的是import注解的解析
  5. 在解析@lmport注解的时候,会有一个getlmports的方法,从主类开始递归解析注解,把所有包含@lmport的注解都解析到,然后在processlmport方法中对Import的类进行分类,此处主要识别的时候AutoConfigurationlmportSelect归属于ImportSelect的子类,在后续过程中会调用deferredlmportSelectorHandler中的process方法,来完整EnableAutoConfiguration的加载。
  6. 通过@Conditional等条件注解按需加载的配置类,其他的将被过滤掉

最后一句话总结:Spring Boot 通过@EnableAutoConfiguration开启自动装配,通过 SpringFactoriesLoader 最终加载META-INF/spring.factories中的自动配置类实现自动装配,自动配置类其实就是通过@Conditional按需加载的配置类,想要其生效必须引入spring-boot-starter-xxx包实现起步依赖

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/12122.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

询问ChatGPT的高质量答案艺术——提示工程指南(更新中……)

目录前言一、提示工程简介二、提示技巧2-1、生成法律文件2-2、添加提示技巧三、角色扮演3-1、智能手机产品描述3-2、添加角色扮演四、标准提示4-1、写一篇有关于新智能手机的评论4-2、添加标准提示、角色提示、种子词提示等等五、示例很少、或者没有示例5-1、生成一个手机配置六…

机器学习中的数学原理——过拟合、正则化与惩罚函数

通过这篇博客&#xff0c;你将清晰的明白什么是过拟合、正则化、惩罚函数。这个专栏名为白话机器学习中数学学习笔记&#xff0c;主要是用来分享一下我在 机器学习中的学习笔记及一些感悟&#xff0c;也希望对你的学习有帮助哦&#xff01;感兴趣的小伙伴欢迎私信或者评论区留言…

Docker系列 基于OpenAI API自建ChatGPT

转自我的博客文章https://blognas.hwb0307.com/linux/docker/4201&#xff0c;内容更新仅在个人博客可见。欢迎关注&#xff01; 前言 我用帐号/密码使用chatGPT已经有一段时间。但是&#xff0c;我有几个私交较密的朋友&#xff0c;他们并不具备使用chatGPT的条件&#xff1b…

【无功优化】基于多目标差分进化算法的含DG配电网无功优化模型【IEEE33节点】(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

java遍历字符串的方法

在 java中&#xff0c;我们需要遍历字符串&#xff0c;如何遍历呢&#xff1f;首先我们先了解一下遍历的概念&#xff1a; 在我们的计算机中&#xff0c;存储的都是二进制数据&#xff0c;为了方便存储和管理&#xff0c;我们把一段数据分成多个字符串。在 java中&#xff0c;遍…

网络IO(non-blocking io)基础

BIO&#xff08;blocking io&#xff09; 传统的网络io模式&#xff0c;面向流&#xff0c;一个线程对接一个会话&#xff0c;因此高并发时会因线程阻塞而性能低效 Java代码&#xff1a; public class BIO implements Connector {private Integer port 8080;Overridepublic v…

SAP Business Technology Platform (BTP)的架构理解

查资料看到的&#xff0c;转一下&#xff0c;附上链接&#xff1a; SAP Business Technology Platform (BTP)的架构理解 长期以来&#xff0c;我在与客户和伙伴的沟通交流中发现大家依然对SAP业务技术平台 – SAP Business Technology Platform (以下简称BTP)纯有各种疑惑&…

Web 攻防之业务安全:密码找回流程绕过测试.(利用链接跳到后面去)

Web 攻防之业务安全&#xff1a;密码找回流程绕过测试 业务安全是指保护业务系统免受安全威胁的措施或手段。广义的业务安全应包括业务运行的软硬件平台&#xff08;操作系统、数据库&#xff0c;中间件等&#xff09;、业务系统自身&#xff08;软件或设备&#xff09;、业务所…

13.vue-cli

单页面应用程序&#xff1a;所有的功能只在index.html中完成 vue-cli是vue版的webpack 目录 1 安装vue-cli 2 创建项目 3 使用预设 4 删除预设 5 开启项目 6 项目文件内容 6.1 node_moduls 中是项目依赖的库 6.2 public 6.2.1 favicon.ico 是浏览器页签内部…

Android屏幕适配dp、px两套解决办法

最新最全文章(2018-08-25)&#xff1a;Android dp方式的屏幕适配-原理(后期补充完整讲解)_手机dp输出是横屏还是竖屏_android阿杜的博客-CSDN博客 “又是屏幕适配&#xff0c;这类文章网上不是很多了吗&#xff1f;” 我也很遗憾&#xff0c;确实又是老问题。但本文重点对网上…

MYSQL学习 - DDL数据库操作

前言 从今天开始, 健哥就带各位小伙伴学习数据库技术。数据库技术是Java开发中必不可少的一部分知识内容。也是非常重要的技术。本系列教程由浅入深, 全面讲解数据库体系。 非常适合零基础的小伙伴来学习。 ------------------------------前戏已做完&#xff0c;精彩即开始---…

基于springboot和ajax的简单项目 06 日志界面的delete功能(根据选择的checkbox)

01.这次后台开始&#xff1b; 顺序依次是dao->xml->service->serviceimpl->controller->html 02.dao接口 public int doDeleteObjects(Param("ids") Integer... ids);03.xml文件 <update id"doDeleteObjects" >delete from sys_lo…

用友U8 cloud,信创云ERP的数智先锋

编辑&#xff1a;阿冒设计&#xff1a;沐由从来未曾有过一个春天&#xff0c;能够像当下这般被如此由衷地期待。经历了漫长的1000多个日日夜夜之后&#xff0c;我们的工作与生活终于回到正轨。自2023年以来&#xff0c;中国市场迎来“开门红”&#xff0c;消费市场加速回暖&…

第十四届蓝桥杯大赛软件赛省赛 C/C++ 大学 A 组 E 题

颜色平衡树问题描述格式输入格式输出样例输入样例输出评测用例规模与约定解析参考程序问题描述 格式输入 输入的第一行包含一个整数 n &#xff0c;表示树的结点数。 接下来 n 行&#xff0c;每行包含两个整数 Ci , Fi&#xff0c;用一个空格分隔&#xff0c;表示第 i 个结点 …

新能源汽车高压配电管理(PDU/BDU)

一、概念与组成 PDU(Power Distribution Unit)&#xff0c;即高压配电单元&#xff0c;功能是负责新能源车高压系统中的电源分配与管理&#xff0c;为整车提供充放电控制、高压部件上电控制、电路过载短路保护、高压采样、低压控制等功能&#xff0c;保护和监控高压系统的运行…

智慧井盖-物联网智能井盖系统-管网数字化监测,守护城市生命线

平升电子智慧井盖-物联网智能井盖系统-管网数字化监测,守护城市生命线实现对井下设备和井盖状态的监测及预警&#xff0c;是各类智慧管网管理系统中不可或缺的重要设备&#xff0c;解决了井下监测环境潮湿易水淹、电力供应困难、通讯不畅等难题&#xff0c; 适合安装于城市主干…

【MySQL--05】表的约束

文章目录 1.表的约束1.1空属性1.2默认值default vs null1.3列描述1.4 zerofill1.5主键primary key1.6 自增长auto_increment1.7唯一键 unique如何设计主键&#xff1f;1.8 外键 foreign key 1.表的约束 真正的约束字段的是数据类型&#xff0c;但是数据类型约束很单一&#xf…

基于springboot和ajax的简单项目 02.一直会出现的页面的上一页,下一页,分页与总页数 (下)

在各种功能中会一直出现页面分页的问题。 对此&#xff0c;可以使用pojo对象&#xff0c;来一直管理页面分页的功能。 01.创建相关的pojo对象。 由于属性是来辅助sql语句的&#xff0c;这个pojo对象。 Setter Getter ToString NoArgsConstructorpublic class PageObject<T&…

day11_面向对象

今日内容 零、 复习昨日 一、作业 二、局部变量&成员变量 三、this关键字 四、构造方法 五、重载 零、 复习昨日 晨考 public class Phone {// 成员属性/成员变量// 数据类型 变量名;double price;String brand;// 成员方法public void call(String num) {System.out.print…

流程引擎基础知识

流程引擎基础知识 流程部署流程取消部署流程发起流程取回流程作废流程委托流程流转常用流程表介绍备注 流程部署 1.后台直接导入bpmn /**流程部署源代码*/public void deploy() {ProcessEngine processEngine ProcessEngines.getDefaultProcessEngine();RepositoryService re…