Spring循环依赖的成因与破局

 一、Spring注入类型

        Spring 核心功能之一依赖注入,依赖注入是使用 Spring 框架的基本手段,通过他获取各种类型的 bean,但使用不同的依赖注入类型时经常会遇到循环依赖的问题。Spring 依赖注入类型:

  1. 字段注入,这是最常用的方式,使用简单方便。但它确是 3 种方式中应该避免的,可能导致潜在的循环依赖。Spring 官方也不推荐使用这种方式,而是推荐构造器注入。
  2. 构造器注入
  3. Setter 方法注入,这种方式可以解决循环依赖

二、创建 Bean 过程

        首先看下 Spring 创建 bean 的过程

        Spring 获取到 Class 文件后并没有直接创建对象,而是先获取对象的信息,将对象的信息保存到 BeanDefinition 中,BeanDefinition 中包含了诸多信息,比如是否懒加载、单例或者多例、是否抽象、是否为 Primary 等等属性,更多详细属性可参考源码。定义好后将这些信息保存到 map 中,这时还需要判断是否需要对用户的 Bean 进行扩展,有有扩展需求,需要对 map 中的对象进行 update。

        对象实例后被保存到了缓存中,实际上就是被保存到了 map 里,单例模式下需要获取时直接从缓存中获取,如下图:

三、循环依赖

        在 Spring 框架这样的依赖注入(Dependency Injection,DI)框架中,循环依赖指的是两个或多个对象之间相互依赖,形成一个闭环结构,即 A 对象依赖于 B 对象,B 对象又反过来依赖于A 对象,或者通过一系列间接依赖构成闭合的依赖链条。

        那Spring是如何解决循环依赖的呢?下面我们分析一下。

        首先需要确认一个问题,Spring是允许循环依赖,在在源码中也有所体现。

/** Whether to automatically try to resolve circular references between beans. */
private boolean allowCircularReferences = true;

        在源码中有这样一个属性,allowCircularReferences 的默认值为 true,也就是说,Spring自身的设计上是允许循环依赖的。

        为了解决循环依赖,Spring 引入了三级缓存,第一级缓存用来持有完整的 Bean 实例,也就是上文提到的单例模式下的缓存;第二级缓存中存放的是提前暴露的对象,已经创建还未完成属性注入;第三级缓存用来存放工厂对象。

@Nullable
protected Object getSingleton(String beanName, boolean allowEarlyReference) {// singletonObjects 为一级缓存Object singletonObject = this.singletonObjects.get(beanName);if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {synchronized (this.singletonObjects) {// earlySingletonObjects 为二级缓存singletonObject = this.earlySingletonObjects.get(beanName);if (singletonObject == null && allowEarlyReference) {// singletonFactories 为三级缓存ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);if (singletonFactory != null) {singletonObject = singletonFactory.getObject();this.earlySingletonObjects.put(beanName, singletonObject);this.singletonFactories.remove(beanName);}}}}return singletonObject;
}

        获取对象的流程,首先是从一级缓存中获取对象,如果获取不到再从二级缓存中获取,如果还是获取不到,则从三级缓存中获取对象工厂,然后通过工厂获取,获取成功后就把对象从三级缓存移动到二级缓存,从而为下一次对象获取做准备。

        解决循环依赖的关键是点还是在 Bean 的生命周期上,通过 @Autowire 注入 Bean 时是在doCreateBean() 方法中完成创建的,这个方法包括三个核心步骤,首先通过 createBeanInstance 实例化 bean,然后通过 populateBean() 方法实现属性注入,最后通过initializeBean() 对 bean 进行扩展。第一步完成了 bean 的初始化,第二步完成了 bean 的完成实例化。

        3.1 循环依赖解决分析                

        假设先初始化 ClassA,在通过 createBeanInstance 方法创建了实例,并将这个实例提前暴露到三级缓存中,然后 ClassA 通过 populateBean 方法进行属性注入,发现自己依赖 ClassB,就会尝试获取 ClassB 实例,显然 ClassB 还没有被创建,所以走创建流程,ClassB 在初始化的第一个流程发现自己依赖 ClassA,就会尝试从第一级缓存获取 ClassA 实例,因为 ClassA 还没有完全创建完毕,所以一级缓存不存在,同样二级缓存中也不存在,当尝试访问第三级缓存时,ClassA 已经提前暴露出去了,所以 ClassB 能从三级缓存中获取到 ClassA 对象并顺利完成所有初始化流程,ClassB 创建完成后,会把自己放到一级缓存中,这时 ClassA 就能从一级缓存中获取到 ClassB,进而完成 ClassA 的初始化。

        到这里应该能理解构造器无法解决循环依赖了吧。因为构造器注入发生在创建bean的第一个步骤,而这个步骤还没有完成三级缓存的构建,自然无法获取到目标对象。

        总结来说,循环依赖是一种不利于软件设计和维护的关系,因为它可能导致初始化顺序混乱、资源占用过多等问题,因此在设计和编码时应尽量避免。在 Spring 框架中,虽然对 setter 注入的循环依赖进行了某种程度的支持,但仍建议遵循良好的设计原则,例如依赖倒置原则和单一职责原则,以减少循环依赖的可能性。

往期经典推荐

SpringBoot项目并发处理大揭秘,你知道它到底能应对多少请求洪峰?_springboot并发处理-CSDN博客

一文看懂Nacos如何实现高效、动态的配置中心管理-CSDN博客

跨越微服务边界:Spring Cloud Sleuth 如何助力实现无缝分布式追踪-CSDN博客

Kafka消息流转的挑战与对策:消息丢失与重复消费问题_kafka发送消息生产者关闭了-CSDN博客

决胜高并发战场:Redis并发访问控制与实战解析-CSDN博客

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Word中解决插入脚注导致的分页位置错误问题

先放一个截图&#xff1a; 上面的截图中&#xff0c;样式为标题3的段落“四、固执的念头”前插入了连续型分节符&#xff0c;并且该分节符的样式为正文&#xff0c;前后的正文段落中有脚注&#xff0c;结果在分页时&#xff0c;标题3段落“四、固执的念头”后的正文段落自动进入…

什么台灯对眼睛好?揭秘四款央视推荐的护眼台灯

近年来&#xff0c;随着电子产品的普及&#xff0c;虽说给生活带来了许多便利&#xff0c;不过对于眼睛还没发育完全的孩子而言&#xff0c;经常使用电子产品是非常容易伤眼的&#xff0c;更何况这些孩子每天还需要长时间的用眼学习&#xff0c;眼睛的负担是非常大的。所以在学…

谷粒商城【成神路】-【10】——缓存

目录 &#x1f9c2;1.引入缓存的优势 &#x1f953;2.哪些数据适合放入缓存 &#x1f32d;3.使用redis作为缓存组件 &#x1f37f;4.redis存在的问题 &#x1f9c8;5.添加本地锁 &#x1f95e;6.添加分布式锁 &#x1f95a;7.整合redisson作为分布式锁 &#x1f697…

java学习之路-数据类型与变量

目录 数据类型与变量 1. 字面常量 2. 数据类型 3. 变量 3.1 变量概念 3.2 整型变量 3.2.1 整型变量 3.2.2 长整型变量 3.2.3 短整型变量 3.2.4 字节型变量 3.3 浮点型变量 3.3.1 双精度浮点型 3.3.2 单精度浮点型 3.4 字符型变量 3.5布尔型变量 3.6 类型转换 …

MySQL--索引优化实战篇(2)

前言&#xff1a; 我们常说的SQL优化&#xff0c;简单来说就是索引优化&#xff0c;通过合理创建索引&#xff0c;调整SQL语法等&#xff0c;来提升查询效率&#xff0c;想要进行SQL优化&#xff0c;就必须知道索引的原理&#xff0c;而且能够看懂SQL的执行计划。 MySQL–索引…

数据保护设备的主要功能是什么

数据保护设备在当今数字化时代扮演着至关重要的角色。随着信息技术的迅猛发展&#xff0c;数据的产生、传输和存储量呈现出爆炸式增长&#xff0c;数据的安全性和完整性成为了企业和个人关注的重点。数据保护设备作为保障数据安全的重要手段&#xff0c;正逐渐受到广泛关注和应…

Python图像处理【22】基于卷积神经网络的图像去雾

基于卷积神经网络的图像去雾 0. 前言1. 渐进特征融合网络2. 图像去雾2.1 网络构建2.2 模型测试 小结系列链接 0. 前言 单图像去雾 (dehazing) 是一个具有挑战性的图像恢复问题。为了解决这个问题&#xff0c;大多数算法都采用经典的大气散射模型&#xff0c;该模型是一种基于单…

基于多源信息融合的巡飞弹对地目标识别与毁伤评估

源自&#xff1a;系统仿真学报 作者&#xff1a;徐艺博 于清华 王炎娟 郭策 冯世如 卢惠民 “人工智能技术与咨询” 发布 摘 要 面向利用多枚巡飞弹对地面高防御移动目标进行打击的任务场景&#xff0c;提出一种基于多源信息融合的巡飞弹对地移动目标识别与毁伤评估方法…

Vue2利用创建a标签实现下载本地静态文件到本地电脑上的功能

最近PC项目遇到一个需求&#xff0c;那就是需要前端下载前端代码包里的前端文件到本地&#xff0c;并且可以给下载下来的文件名指定任意的文件名&#xff0c;如下图所示&#xff0c;在前端代码里public里的statics里有个静态文件zswj.pem&#xff0c;页面上有个下载按钮&#x…

基于.Net 的图形验证码模块

&#x1f3c6;作者&#xff1a;科技、互联网行业优质创作者 &#x1f3c6;专注领域&#xff1a;.Net技术、软件架构、人工智能、数字化转型、DeveloperSharp、微服务、工业互联网、智能制造 &#x1f3c6;欢迎关注我&#xff08;Net数字智慧化基地&#xff09;&#xff0c;里面…

ARM TrustZone技术解析:构建嵌入式系统的安全扩展基石

&#x1f308;个人主页: Aileen_0v0 &#x1f525;热门专栏: 华为鸿蒙系统学习|计算机网络|数据结构与算法|MySQL| ​&#x1f4ab;个人格言:“没有罗马,那就自己创造罗马~” #mermaid-svg-dSk2aQ85ZR0zxnyI {font-family:"trebuchet ms",verdana,arial,sans-serif;f…

【Python】科研代码学习:七 TrainingArguments,Trainer

【Python】科研代码学习&#xff1a;七 TrainingArguments&#xff0c;Trainer TrainingArguments重要的方法 Trainer重要的方法使用 Trainer 的简单例子 TrainingArguments HF官网API&#xff1a;Training 众所周知&#xff0c;推理是一个大头&#xff0c;训练是另一个大头 之…