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

news2024/11/15 12:23:55

文章目录

  • 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包实现起步依赖

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

《SQLi-Labs》01. Less 1~5

Less-1 ~ Less-5 前言Less-1知识点题解 Less-2题解 Less-3题解 Less-4题解 Less-5知识点题解 sqli。开启新坑。 前言 对于新手&#xff0c;为了更加直观的看到 sql 注入语句&#xff0c;可以在以下文件添加两句&#xff1a; echo $sql; # 将构造的 sql 语句进行输出 echo &qu…

Maven依赖冲突分析和解决

使用maven管理jar包依赖时&#xff0c;可能会出现jar包版本冲突&#xff0c;不同版本的api调用方式可能不同&#xff0c;会出现NoSuchMethodError和ClassNotFoundException问题&#xff0c;甚至编译不通过&#xff0c;如&#xff1a;在common-lang3 的3.8.1版本中MethodUtils::…

Python中的@cache巧妙用法

缓存是一种空间换时间的策略&#xff0c;缓存的设置可以提高计算机系统的性能&#xff0c;这篇文章主要介绍了Python中的cache巧妙用法,需要的朋友可以参考下 Python中的cache有什么妙用&#xff1f; 缓存是一种空间换时间的策略&#xff0c;缓存的设置可以提高计算机系统的性…

Spark 简介与原理

目录标题1 Spark 简介与原理1.1 Spark与Hadoop的区别1.2 Spark的应用场景1.3 Spark的作业运行流程1.4 Spark 2.X与Spark 1.X的区别1 Spark 简介与原理 Spark 是一个大规模数据处理的统一分析引擎。 具有迅速、通用、易用、支持多种资源管理器的特点。 Spark生态系统: Spark SQL…

双榜加冕!加速科技荣登2023准独角兽中国未来独角兽双榜单

4月10日至11日&#xff0c;由杭州市人民政府、民建浙江省委会、中国投资发展促进会主办的第7届万物生长大会在杭州国际博览中心隆重举行。会上&#xff0c;中国投资发展促进会创投专委会、杭州市创业投资协会联合微链共同发布2023杭州市独角兽&#xff08;准独角兽&#xff09;…

【高危】vm2 <3.9.16 沙箱逃逸漏洞(CVE-2023-29199)

漏洞描述 vm2 是一个基于 Node.js 的沙箱环境&#xff0c;可以使用列入白名单的 Node 内置模块运行不受信任的代码。 vm2 3.9.16之前版本中&#xff0c;由于transformer.js中transformer函数中异常处理逻辑不够完善&#xff0c;攻击者可通过制造异常绕过handleException()并造…

精通 TensorFlow 2.x 计算机视觉:第一部分

原文&#xff1a;Mastering Computer Vision with TensorFlow 2.x 协议&#xff1a;CC BY-NC-SA 4.0 译者&#xff1a;飞龙 本文来自【ApacheCN 深度学习 译文集】&#xff0c;采用译后编辑&#xff08;MTPE&#xff09;流程来尽可能提升效率。 不要担心自己的形象&#xff0c;…

Downie 4 4.6.14 MAC上最新最好用的一款视频下载工具

Downie for Mac 简介 Downie是Mac下一个简单的下载管理器&#xff0c;可以让您快速将不同的视频网站上的视频下载并保存到电脑磁盘里然后使用您的默认媒体播放器观看它们。 Downie 4 下载 Downie 4 for Mac Downie 4 for Mac软件特点 支持许多站点 -当前支持1000多个不同的…

printf里的格式控制符

%p&#xff1a;打印地址(指针地址)&#xff0c;十六进制形式输出&#xff0c;有多少位输出多少位&#xff0c;取决于是32 or 64位系统&#xff0c;如果输出不够位宽&#xff0c;左边补0。 &#xff05;f用来输出实数&#xff0c;以小数形式输出&#xff0c;默认情况下保留小数点…

自动化测试工程师需要具备什么技能?

如果是初入门的学习者&#xff0c;不建议拿一本书从头学&#xff0c;很可能会被里边一些专业术语和不常用的技术带偏&#xff0c;不论在公司还是在其他岗位上自学测试&#xff0c;都可以用自己搭建好的项目来练手&#xff08;如果在公司有现成的项目更好&#xff09;&#xff0…

提取图像特征方法总结 是那种很传统的方法~

目录 写在前面 一、SIFT&#xff08;尺度不变特征变换&#xff09; 1.SIFT特征提取的实质 2.SIFT特征提取的方法 3.SIFT特征提取的优点 4.SIFT特征提取的缺点 5.SIFT特征提取可以解决的问题&#xff1a; 二、HOG&#xff08;方向梯度直方图&#xff09; 1.HOG特征提取…

30岁软件测试,目前已失业4个月,迷茫不知该怎么办?

本人14年一本毕业&#xff0c;但是人特别懒&#xff0c;不爱学习&#xff0c;专业不好&#xff0c;毕业前都没找到合适工作&#xff0c;直接去创业了&#xff0c;奶茶店&#xff0c;托管&#xff0c;都弄过&#xff0c;也干过销售&#xff0c;反正浑浑噩噩度过了两年&#xff0…

CTFHub | 文件头检查

0x00 前言 CTFHub 专注网络安全、信息安全、白帽子技术的在线学习&#xff0c;实训平台。提供优质的赛事及学习服务&#xff0c;拥有完善的题目环境及配套 writeup &#xff0c;降低 CTF 学习入门门槛&#xff0c;快速帮助选手成长&#xff0c;跟随主流比赛潮流。 0x01 题目描述…

关于运行时内存数据区的一些扩展概念

栈顶缓存技术&#xff08;Top-of-Stack Cashing&#xff09; 前面提过&#xff0c;基于栈式架构的虚拟机所使用的零地址指令更加紧凑&#xff0c;但完成一项操作的时候必然需要使用更多的入栈和出栈指令&#xff0c;这同时也就意味着将需要更多的指令分派(instruction dispatc…

跨平台科学应用程序:QtiPlot 1.X Crack

QtiPlot 是一个用于数据分析和可视化的跨平台科学应用程序。由于其多语言支持&#xff0c;QtiPlot 被积极用于世界各地学术机构的教学。许多研究科学家信任 QtiPlot 来分析他们的数据并发布他们的工作结果。来自各个科学领域和行业的数千名注册用户已经选择了 QtiPlot 来帮助他…

Part-aware attention correctness for video salient object detection笔记总结

一、摘要 问题&#xff1a;在以往的VSOD中&#xff0c;一般主要是研究时空结构&#xff0c;利用隐式注意力模型去聚合相邻视频帧的互补信息。但很少有方法去关注跨视频帧的亲和力&#xff0c;即建立显式注意力图式去完成VSOD。 解决&#xff1a;提出一个新的注意力正确性策略去…

博客系统测试报告【可上线】

目录 1、测试概述 1.1、项目名称 1.2、测试时间 1.3、编写目的 1.4、测试范围 2、测试计划 2.1、测试用例 2.1.1、注册/登录模块 2.1.2、个人中心模块 2.1.3、找回密码模块 2.1.4、博客主列表模块 2.1.5、个人博客列表模块 2.1.6、个人草稿列表模块 2.1.7、博客详…

Elasticsearch:位置搜索介绍

在这个互联网和信息时代&#xff0c;在应用程序和应用程序中启用基于位置的搜索是一个普遍的要求。 基于位置的搜索根据邻近度获取场所或地点&#xff0c;例如附近的餐馆、半径不超过 1 公里的待售房屋等。 我们还使用基于位置的搜索来查找前往某个地方或兴趣点的方向。 好消息…

C#代码修改设计原图psd、ai格式图层文字内容等导出bmp等,需要license 要高额付费放弃

Update text is supported only in licensed mode System.ComponentModel.LicenseException HResult0x80131901 MessageUpdate text is supported only in licensed mode SourceAspose.PSD StackTrace: at  .(Object ) at  . () at  .(Object , UInt32 )…

【C++学习】类和对象--对象特性

构造函数和析构函数 对象的初始化和清理是两个非常重要的安全问题 一个对象或变量没有初始状态&#xff0c;对其使用后果是未知的 使用完一个对象或变量&#xff0c;没有及时清理&#xff0c;也会造成一定的安全问题 C利用构造函数和析构函数解决上述问题&#xff0c;这两个函数…