高级java每日一道面试题-2025年01月13日-框架篇[Spring篇]-Spring 是怎么解决循环依赖的?

server/2025/1/15 11:47:30/

如果有遗漏,评论区告诉我进行补充

面试官: Spring 是怎么解决循环依赖的?

我回答:

在Java高级面试中,Spring框架如何解决循环依赖是一个重要且常见的问题。以下是对Spring解决循环依赖的详细解释:

循环依赖的定义与类型

循环依赖是指两个或多个Bean之间互相依赖,形成一个闭环。在Spring框架中,循环依赖通常发生在依赖注入(Dependency Injection)过程中。循环依赖的类型主要包括构造函数循环依赖和属性(或Setter)循环依赖。

Spring 如何检测和解决循环依赖?

Spring 使用了多种策略来检测并解决循环依赖问题,主要包括以下几种方式:

单例 Bean 的三级缓存机制

Spring 容器为每个单例 Bean 维护了三个不同级别的缓存:

  • singletonObjects:存放已经完全初始化完成的 Bean 实例。
  • earlySingletonObjects:存放尚未完成初始化但已经被实例化的 Bean(即半成品 Bean)。这些 Bean 已经完成了构造函数注入,但是还没有完成属性注入和其他初始化方法调用。
  • singletonFactories:存放 FactoryBean 或者用于创建早期暴露对象的工厂方法。

当 Spring 检测到 A 和 B 之间的循环依赖时,它会按照如下步骤操作:

  1. 创建 Bean A:Spring 开始创建 Bean A,并将其放入 singletonFactories 中。
  2. 实例化 Bean A:接着实例化 Bean A 并设置其非循环依赖的属性。
  3. 提前暴露 Bean A:将 Bean A 提前暴露给其他 Bean 使用,此时 Bean A 被移入 earlySingletonObjects 缓存。
  4. 创建 Bean B:尝试创建 Bean B,在此过程中发现它依赖于 Bean A。
  5. 获取 Bean A:由于 Bean A 已经存在于 earlySingletonObjects 中,所以可以直接获取并设置到 Bean B 上。
  6. 继续初始化 Bean A:回到 Bean A 的初始化流程,设置它的剩余属性(包括对 Bean B 的引用),最后将 Bean A 移入 singletonObjects 缓存。
  7. 完成 Bean B 的初始化:现在可以安全地完成 Bean B 的初始化,因为它已经获得了对 Bean A 的引用。
    通过这种方式,Spring 成功地打破了循环依赖,使得两个相互依赖的 Bean 都能正常初始化。
构造器注入 vs 字段/Setter 注入
  • 构造器注入:如果使用构造器注入,则在构造函数中传递所有必需的依赖项。在这种情况下,如果存在循环依赖,Spring 将抛出异常,因为无法同时满足两个 Bean 的构造需求。
  • 字段/Setter 注入:相比之下,字段注入或 Setter 方法注入允许 Spring 在实例化之后再设置依赖关系,这为解决循环依赖提供了可能。因此,推荐在需要支持循环依赖的情况下优先考虑字段或 Setter 注入。
@Lazy 注解

对于某些特定场景下的循环依赖问题,可以使用 @Lazy 注解延迟加载某个 Bean,直到真正需要它的时候才进行实例化。这样可以避免在启动阶段就触发循环依赖错误。

java">@Component
public class ServiceA {private final ServiceB serviceB;@Autowiredpublic ServiceA(@Lazy ServiceB serviceB) {this.serviceB = serviceB;}
}
代理模式

Spring 还可以通过 CGLIB 动态代理的方式生成目标 Bean 的代理类,然后在代理类中实现对原始 Bean 的懒加载。这种方式适用于接口类型的 Bean,能够有效缓解循环依赖问题。

Spring无法解决的循环依赖情况

需要注意的是,Spring的循环依赖解决机制有一些限制:

  1. 原型作用域的Bean:对于原型作用域的Bean,由于每次请求都会创建一个新的Bean实例,因此无法使用缓存来解决循环依赖。
  2. 构造器注入的循环依赖:如果Bean的构造方法中存在循环依赖,Spring也无法解决。因为在构造方法中,Bean实例还未创建,无法放入缓存。

解决循环依赖的最佳实践

尽管Spring提供了解决循环依赖的机制,但在设计时仍应尽量避免出现循环依赖,因为循环依赖可能导致代码的可读性差,并且可能是设计上的问题。以下是一些解决循环依赖的最佳实践:

  1. 模块化:将代码拆分成独立的模块,使每个模块只负责一个功能,降低模块间的耦合度。
  2. 使用依赖注入:通过依赖注入,将依赖关系从代码中解耦,使得一个类不再直接依赖另一个类,而是依赖于一个接口或抽象类。
  3. 使用设计模式:利用设计模式(如观察者模式、中介者模式等)来帮助更好地组织代码,避免循环依赖的产生。
  4. 代码重构:定期对代码进行重构,消除潜在的循环依赖问题。

http://www.ppmy.cn/server/158546.html

相关文章

5 list 语法

在 Shell 脚本中,列表(数组)是一种非常有用的数据结构,可以用来存储多个值。 定义数组 # 定义一个空数组 my_array()# 定义一个带有初始值的数组 my_array("value1" "value2" "value3")访问数组元…

k8s故障 ImagePullBackOff状态排错

需看yaml 这个策略是否开启

(蓝桥杯)二维数组前缀和典型例题——子矩阵求和

题目描述 小 A 同学有着很强的计算能力,张老师为了检验小 AA同学的计算能力,写了一个 n 行 m 列的矩阵数列。 张老师问了小 A 同学 k 个问题,每个问题会先告知小 A 同学 4 个数 x1,y1,x2,y2画出一个子矩阵,张老师请小 A同学计算出…

计算机视觉算法实战——手写公式识别(主页有源码)

✨个人主页欢迎您的访问 ✨期待您的三连 ✨ ✨个人主页欢迎您的访问 ✨期待您的三连 ✨ ✨个人主页欢迎您的访问 ✨期待您的三连✨ ​ ​​​​​​​​​​​​​​​​​​ 1. 领域介绍✨✨ 手写公式识别(Handwritten Mathematical Expression Recognition, HME…

Excel如何制作轮班表

Excel如何制作轮班表 1. 概念讲解2. 例子3. 详细讲解3.1 前期准备3.2 人员依次编号3.3 填入日期,和日期编号3.4 Mod函数-填充值班人员编号3.4 Vlookup函数-进行查找人员 操作文档 1. 概念讲解 轮班是指一种工作安排系统,员工每天、每周或每月在不同班次…

初识JAVA-面向对象的三大特征之多态

1. 重温面向对象 面向对象是一种解决问题的思想,它把计算机程序看作是各种对象组合起来的。每个对象都有自己的数据(属性)和行为(方法),主要依靠对象之间的交互来解决和实现问题。Java是一门纯面向对象的语…

【数据仓库】— 5分钟浅谈数据仓库(适合新手)从理论到实践

大家好,我是摇光~ 对于刚进入大数据领域的萌新,且想要在数据分析岗、数据运维岗、数据工程师这些岗位立足,了解数据仓库是必要的,接下来我尽量用通俗易懂的语言让大家了解到数据仓库。 在当今大数据盛行的时代,数据仓…

【网络云SRE运维开发】2025第3周-每日【2025/01/14】小测-【第13章ospf路由协议】理论和实操

文章目录 选择题(10道)理论题(5道)实操题(5道) 【网络云SRE运维开发】2025第3周-每日【2025/01/14】小测-【第12章ospf路由协议】理论和实操 选择题(10道) 在OSPF协议中&#xff0c…