当前位置: 首页 > news >正文

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

相关文章:

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

文章目录SpringBoot自动装配原理1. 从调用SpringApplication构造器方法开始2. 解析启动类4.按需装配4.1 分析dubbo自动装配5. 如果定义自己的starter6. 面试答法SpringBoot自动装配原理 之前面试被问到这个题目&#xff0c;只会答一些spi、AutoConfigration注解、Import之类的&…...

如何在大厂做好架构演进?

1 架构演进的定义 1.1 定义 通过设计新的系统架构(4R)&#xff0c;来应对业务和技术的发展变化。 1.2 关键点 新架构新的复杂度 1.3 目的 应对业务和技术的发展变化后带来新的复杂度。 案例 淘宝去IOE&#xff0c;是因为业务发展大了后&#xff0c;IOE的成本和可控性难…...

减半技术实现求a的n次幂

目录 减半技术实现求a的n次幂 程序设计 程序分析 减半技术实现求a的n次幂 【问题描述】给定两个正整数a和n,采用减半技术求a的n次幂;其中a<100,b<20; 【输入形式】两个整数a,n(a与n中间用空格隔开); 【输出形式】一个整数 【样例输入1】2 3 【样例输出1】8 【样…...

MYSQL8窗口函数

MYSQL8窗口函数 MYSQL8窗口函数窗口函数分类序号函数--排行榜row_number()示例rank()示例dense_rank()示例partition by对每个分区内的行进行排名不加partition by全局排序 开窗聚合函数分布函数CUME_DIST()PERCENT_RANK() 前后函数LAG()的用法LEAD() 头尾函数其他函数NTH_VALU…...

全国大学生智能汽车竞赛——安装Ubuntu操作系统(双系统)

1.1 电脑分区 1.1.1 分区原因 由于我们想要在电脑上同时安装Windows和Ubuntu系统&#xff0c;所以就要在window使用的内存中划分出来一段用来给Ubuntu系统使用&#xff0c;相当于一个应用程序一样 1.1.2 分区步骤 1.右击此电脑&#xff0c;点击管理&#xff0c;然后双击左侧…...

[STM32F103C8T6]看门狗

看门狗&#xff1a; 在由单片机构成的微型计算机系统中&#xff0c;由于单片机的工作常常会受到来自外界电磁场的干扰&#xff0c;造 成程序的跑飞&#xff0c;而陷入死循环&#xff0c;程序的正常运行被打断&#xff0c;由单片机控制的系统无法继续工作&#xff0c;会 造成整个…...

浪潮:2022年净利同比增长51.39%

一、4月头条 华为的紧急回应&#xff0c;让东方材料21亿收购要黄&#xff1f; 4月10日消息&#xff0c;东方材料昨日晚间公告拟定增募资不超20亿元&#xff0c;用于向诺基亚全资子公司NSN收购TD TECH 51%股权&#xff08;交易对价21.22亿元&#xff09;。TD TECH剩余49%股权由…...

大厂面试内幕:阿里内部整理出的5000页Java面试复盘指南,起飞!!!

互联网的技术岗一直是高薪的代名词&#xff0c;特别是大厂&#xff0c;应届生的年薪基本都20W起&#xff0c;比一般的公司高多了。 看下面这张网上热传的大厂应届生薪酬表就知道了&#xff0c;SP offer甚至能拿到30W以上。 技术社区也有晒出高薪offer的同学&#xff1a; 除了薪…...

数据结构——哈希表相关题目

数据结构——哈希表相关题目 242. 有效的字母异位词1.暴力解法2.排序后比较3.哈希表 383. 赎金信哈希解法 49. 字母异位词分组438. 找到字符串中所有字母异位词3. 无重复字符的最长子串76. 最小覆盖子串349. 两个数组的交集1.排序双指针2.哈希表 350. 两个数组的交集 II1.排序双…...

域名解析设置方法

域名解析设置都是实时生效的&#xff0c;一般只需几秒即可同步到各地 DNS 上&#xff0c;但各地 DNS 均有缓存机制&#xff0c;解析的最终生效取决于各运营商刷新时间! 一、A记录 ①.主机名必须填写&#xff1b; 常用主机名有&#xff1a;www//*&#xff0c;效果参见上图说明&…...

MySQL连接空闲时间超过8小时报错原因与延伸知识

1 错误原因 1.1 两个参数 MySQL服务端两个参数控制连接超时时间&#xff1a; wait_timeoutinteractive_timeout1.1.1 如何查看 show global variables like interactive_timeout show global variables like wait_timeout 复制代码 1.1.2 含义与区别 wait_timeout&#xf…...

Flutter渲染原理

一 Widget Element RenderObject 之间的关系 1 Widget 在Flutter 中&#xff0c;万物皆是Widget,无论是可见的还是功能型的。一切都是Widget. 官方文档中说的Widget 使用配置和状态来描述View 界面应该长什么样子。 它不仅可以表示UI元素&#xff0c;也可以表示一些功能性的…...

PathCore:IAD文献解读

论文链接&#xff1a;[Towards Total Recall in Industrial Anomaly Detection]Towards Total Recall in Industrial Anomaly Detection &#xff1a;数据集&#xff0c; &#xff1a;标签 : 在ImageNet上预训练后的网络 第 张图 网络中第 层 1. Locall…...

C语言判断一个日期是在该年的第几天案例讲解

今天是2023年4月11号&#xff0c;我们就用今天举例得出是2023年的第几天。 思路分析 1&#xff09;我们想知道2023年4月11号是2023年的第几天&#xff0c;只需要把1到3月份的天数累加求和然后加上今天日期也就是11就可以算出2023年4月11号是2023年的第几天。 推广&#xff1a;…...

【超全总结】集成环信消息推送注意事项(华为、oppo、vivo等)

环信即时通讯 IM 支持集成第三方厂商的消息推送服务&#xff0c;为 Android 开发者提供低延时、高送达、高并发、不侵犯用户个人数据的离线消息推送服务。 当客户端应用进程被关闭等原因导致用户离线&#xff0c;环信即时通讯 IM 服务会通过第三方厂商的消息推送服务向该离线用…...

C++回调函数以及epoll中回调函数的使用

回调函数是一种常用的编程技术&#xff0c;它允许程序在运行时将一个函数作为参数传递给另一个函数&#xff0c;以实现更加灵活和可扩展的功能。在C中&#xff0c;回调函数通常被实现为函数指针或者函数对象。 函数指针是指向函数的指针变量&#xff0c;可以通过它来调用函数。…...

0基础学习软件测试有哪些建议

其实现在基础的资料和视频到处都是&#xff0c;就是看你有没有认真的去找学习资源了&#xff0c;去哪里学习都是要看你个人靠谱不靠谱&#xff0c;再好的教程和老师&#xff0c;你自己学习不进去也是白搭在正式选择之前&#xff0c;大可以在各种学习网站里面找找学习资源先自己…...

MySQL数据类型

文章目录一、数据类型分类二、数值类型1. tinyint 类型2. bit 类型3. int 类型4. float 类型5. decimal 类型三、字符串类型1. char 类型2. varchar 类型3. char 和 varchar 比较4.日期和时间类型5. enum 和 set一、数据类型分类 数据类型本质也是一种约束&#xff01; 如果插入…...

【设计模式】从Mybatis源码中学习到的10种设计模式

文章目录 一、前言二、源码&#xff1a;学设计模式三、类型&#xff1a;创建型模式1. 工厂模式2. 单例模式3. 建造者模式 四、类型&#xff1a;结构型模式1. 适配器模式2. 代理模式3. 组合模式4. 装饰器模式 五、类型&#xff1a;行为型模式1. 模板模式2. 策略模式3. 迭代器模式…...

爬虫攻守道 - 猿人学第20题 - 殊途同归

写在开头 这题也是&#xff0c;自己搞顶多追踪到wasm代码&#xff0c;然后就走不下去了。找了2个参考方案&#xff0c;自己做的过程中还又遇到些新的问题&#xff0c;下面做个记录。解法1参考文章解法2参考文章 解法1&#xff1a;追根溯源 在 JS 代码中追踪到 Payload 赋值位…...

4.11日报

synchronized 和 ReentrantLock 区别是什么&#xff1f; synchronized 早期的实现比较低效&#xff0c;对比 ReentrantLock&#xff0c;大多数场景性能都相差较大&#xff0c;但是在 Java 6 中对 synchronized 进行了非常多的改进。 主要区别如下&#xff1a; ReentrantLock …...

【LeetCode每日一题: 1039. 多边形三角剖分的最低得分 | 暴力递归=>记忆化搜索=>动态规划 | 区间dp 】

&#x1f34e;作者简介&#xff1a;硕风和炜&#xff0c;CSDN-Java领域新星创作者&#x1f3c6;&#xff0c;保研|国家奖学金|高中学习JAVA|大学完善JAVA开发技术栈|面试刷题|面经八股文|经验分享|好用的网站工具分享&#x1f48e;&#x1f48e;&#x1f48e; &#x1f34e;座右…...

Okio 网络提速

文章目录网络数据处理流程Page Cache传统 I/O 拷贝的性能问题零拷贝技术DMA 技术零拷贝技术分类mmapsendfilespliceDirect I/O零拷贝技术性能分析小结OkioOkio 的使用Okio 网络提速的原理Okio 总结总结网络数据处理流程 在讲 Okio 之前&#xff0c;为了能更好的了解 Okio 的优…...

自动驾驶企业面临哪些数据安全挑战?

近期&#xff0c;“特斯拉员工被曝私下分享用户隐私”不可避免地成了新闻热点&#xff0c;据说连马斯克也不能幸免。 据相关媒体报道&#xff0c;9名前特斯拉员工爆料在2019年至2022年期间&#xff0c;特斯拉员工通过内部消息系统私下分享了一些车主车载摄像头记录的隐私视频和…...

Doris(2):Doris编译部署

1 Doris编译 Apache Doris提供直接可以部署的版本压缩包&#xff1a;https://cloud.baidu.com/doc/PALO/s/Ikivhcwb5 也可以自行编译压缩包后使用&#xff08;推荐&#xff09; 1.1 使用 Docker 开发镜像编译&#xff08;推荐&#xff09; 这个是官方文档推荐的&#xff0c;…...

使用MyBatis实现简单查询

文章目录一&#xff0c;创建数据库与表&#xff08;一&#xff09;在Navicat里创建MySQL数据库testdb&#xff08;二&#xff09;创建用户表 - t_user&#xff08;三&#xff09;在用户表里插入3条记录二&#xff0c;案例演示MyBatis基本使用&#xff08;一&#xff09;创建Mav…...

C指针(*point)[4]和char *point[4]

char (*point)[4] // 数组指针。 a[3][4] // 先申明二维数组,用它来指向这个二维数组. char *point[4] // 指针数组。 a[4][5] // 一连串的指针. char (*point)[4] // 一个指针&#xff0c;指向有4个元素的数组&#xff1b;占内存大小为 4 个字节 ch…...

【Bard】谷歌的人工智能工具—Bard初体验

文章目录一、Bard介绍二、Bard体验1、加入Bard的候补名单2、登入Bard篇3、使用Bard篇&#xff08;1&#xff09;提供三种预选方式✨&#xff08;2&#xff09;创作生成各类文案&#xff08;3&#xff09;无生成图画能力&#xff08;4&#xff09;支持语音转文本输入✨&#xff…...

2022国赛30:windows脚本题解析

大赛试题内容: ( 九) ) 脚本 【任务描述】 为了减少重复性任务的工作量,节省人力和时间,请采用脚本,实现快速批量的操作。 1.在 windows4 上编写 C:\CreateFile.ps1 的 powershell 脚本,创建20 个文件 C:\test\File00.txt 至 C:\test\File19.txt,如果文件存在,则首先删除…...

Excel常用函数公式20例

目录 一、【IF函数条件判断】 二、【多条件判断】 三、【条件求和】 四、【多条件求和】 五、【条件计数】 六、【多条件计数】 七、【条件查找】 八、【多条件查找】 九、【计算文本算式】 十、【合并多个单元格内容】 十一、【合并带格式的单元格内容】 十二、…...