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博客

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

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

相关文章

酷开科技发力研发酷开系统,让家庭娱乐生活更加丰富多彩

在这个快节奏的社会&#xff0c;家庭娱乐已成为我们日常生活中不可或缺的一部分&#xff0c;为了给家庭带来更多欢笑与感动&#xff0c;酷开科技发力研发出拥有丰富内容和技术的智能电视操作系统——酷开系统&#xff0c;它集合了电影、电视剧、综艺、游戏、音乐等海量内容&…

01-分析同步通讯/异步通讯的特点及其应用

同步通讯/异步通讯 微服务间通讯有同步和异步两种方式 同步通讯: 类似打电话场景需要实时响应(时效性强可以立即得到结果方便使用),而且通话期间不能响应其他的电话(不支持多线操作)异步通讯: 类似发邮件场景不需要马上回复并且可以多线操作(适合高并发场景)但是时效性弱响应…

Unmanaged PowerShell

简介 在渗透测试当中经常会使用到PowerShell来执行脚本, 但是直接使用PowerShell.exe是一个非常敏感的行为, EDR等产品对PowerShell.exe进程的创建监控的很密切, 并且随着PowerShell的渗透测试工具的普及, 越来越多的EDR会利用微软提供的AMSI接口对PS脚本进行扫描, 但是对于低…

[短文]不同空白字符导致程序执行失败问题

屏幕显示的一个空白字符&#xff0c;对于编程者来说&#xff0c;并无差异&#xff0c;但底层截然不同的表示方法&#xff0c;极大可能导致程序执行失败&#xff01; 今天博主就遇到一个空格字符的问题&#xff0c;大概情况是前端编写SQL传入&#xff0c;后端有时可以执行&…

Uninty 鼠标点击(摄像机发出射线-检测位置)

平面来触发碰撞&#xff0c;胶囊用红色材质方便观察。 脚本挂载到胶囊上方便操作。 目前实现的功能&#xff0c;鼠标左键点击&#xff0c;胶囊就移动到那个位置上。 using System.Collections; using System.Collections.Generic; using UnityEngine;public class c6 : MonoBe…

DNS服务

简介 DNS&#xff1a;domain name service 域名服务 作用&#xff1a;为客户机提供域名解析 dns 监听端口&#xff1a;53端口 分为 udp&#xff08;只负责向外发&#xff09; 和 tcp&#xff08;确保正常发送至对端&#xff0c;对端发送数据包表示已经收到&#xff09; 搭建…

解密QQ盗号诈骗APP:逆向溯源,探寻幕后黑色操作

逆向溯源qq盗号诈骗app 起因 专注于web漏洞挖掘、内网渗透、免杀和代码审计&#xff0c;感谢各位师傅的关注&#xff01;网安之路漫长&#xff0c;与君共勉&#xff01; 分析该app是源于朋友被盗号了&#xff0c;对方发了个app想盗号&#xff0c;这怎么能惯着他&#xff1f;这不…

day16_购物车(添加购物车,购物车列表查询,删除购物车商品,更新选中商品状态,完成购物车商品的全选,清空购物车)

文章目录 购物车模块1 需求说明2 环境搭建3 添加购物车3.1 需求说明3.2 远程调用接口开发3.2.1 ProductController3.2.2 ProductService 3.3 openFeign接口定义3.3.1 环境搭建3.3.2 接口定义3.3.3 降级类定义 3.4 业务后端接口开发3.4.1 添加依赖3.4.2 修改启动类3.4.3 CartInf…

MyBatisPlus理解

MyBatisPlus是mybatis的增强&#xff0c;mybatis是数据库持久化的框架&#xff0c;但mybatisplus并不是替代mybatis&#xff0c;而是相辅相成的关系 MyBatisPlus不会对以前使用mybatis开发的项目进行影响&#xff0c;引入后仍然正常运行。 使用方法&#xff1a; 1.在引入了对…

[力扣 Hot100]Day48 路径总和 III

题目描述 给定一个二叉树的根节点 root &#xff0c;和一个整数 targetSum &#xff0c;求该二叉树里节点值之和等于 targetSum 的 路径 的数目。 路径 不需要从根节点开始&#xff0c;也不需要在叶子节点结束&#xff0c;但是路径方向必须是向下的&#xff08;只能从父节点到…

再谈 Cookie 和 Session

文章目录 1. 核心方法&#xff1a;HttpServletRequest 类中HttpServletResponse 类中HttpSession类中Cookie类中 2.实现登录界面 Cookie 是浏览器在本地持久化存储数据的一种机制。 Cookie 的数据从哪里来&#xff1f; 是服务器返回给浏览器的。 Cookie 的数据长什么啥样&#…

Chapter20-Ideal gases-CIE课本要点摘录、总结

20.1 Particles of a gas Brownian motion Fast modules 速率的数值大概了解下&#xff1a; average speed of the molecules:400m/s speed of sound:approximately 330m/s at STP&#xff08;standard temperature and pressure&#xff09; Standard Temperature and Pres…

不同路径 不同路径 II 整数拆分

62.不同路径 力扣题目链接(opens new window) 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish” &#xff09;。…

标准不锈钢电阻-栅极电阻器的设计方案

EAK不锈钢栅极电阻器 可靠的不锈钢电阻元件和端子 所有装置均为三重绝缘&#xff0c;适用于 1000V 交流或直流 标准电网电阻器有现货&#xff0c;可减少代价高昂的停机时间 在很宽的温度范围内具有稳定的耐受性 高功率可节省空间、重量和成本 标准 26.5 英寸尺寸&#xf…

PDF24 Creator PDF工具箱 v11.17.0

软件介绍 可将大部分文件转成pdf格式的免费软件&#xff0c;安装好后会在你的打印机里看到一个叫PDF24的虚拟打印机&#xff0c;你可将要转成pdf格式的文件打印时选虚拟打印机PDF24&#xff0c;也可以直接将文件以拖拉方式拉进这软件的主视窗编辑区里&#xff0c;它会自动转成…

OD_2024_C卷_200分_6、六_连续出牌数量【JAVA】【回溯算法】

题目描述 package odjava;import java.util.Arrays; import java.util.Scanner;public class 六_连续出牌数量 {// 定义扑克牌类static class Card {int num; // 牌号char color; // 花色public Card(int num, String color) {this.num num;this.color color.charAt(0); // 取…

Liinux——(网络)socket编程

预备知识 源IP地址和目的IP地址 在IP数据包头部中, 有两个IP地址, 分别叫做源IP地址, 和目的IP地址 认识端口号 端口号(port)是传输层协议的内容. 端口号是一个2字节16位的整数;端口号用来标识一个进程, 告诉操作系统, 当前的这个数据要交给哪个进程来处理;IP地址 端口号能…

js【详解】DOM

文档对象模型&#xff08;Document Object Model&#xff0c;简称DOM&#xff09; DOM 是哪种数据结构 &#xff1f; DOM 的本质是浏览器通过HTML代码解析出来的一棵 树。 操作 DOM 常用的 API 有哪些 &#xff1f; 获取 DOM 节点 //方式 1&#xff1a;通过【id】获取&#xf…

每日学习笔记:C++ STL 的队列Deque

定义 内存模型 Deque与Vector比较 操作函数 运用实例

每日OJ题_牛客HJ73 计算日期到天数转换(IO型OJ)

目录 牛客HJ73 计算日期到天数转换 解析代码 牛客HJ73 计算日期到天数转换 计算日期到天数转换_牛客题霸_牛客网 解析代码 #include <iostream> using namespace std; int main() {int year 0, month 0, day 0, sum 0;cin >> year >> month >>…