Spring面试题-Spring是怎么解决循环依赖的?

Spring 使用了三级缓存来解决单例模式下的 setter 循环依赖问题。这三级缓存分别是:

一级缓存(singletonObjects):用于存放完全初始化好的单例 Bean。

二级缓存(earlySingletonObjects):用于存放部分初始化完成的单例 Bean,这些 Bean 已经完成了实例化但尚未完成属性填充和初始化。这个缓存主要用于解决循环依赖问题。

三级缓存(singletonFactories):用于存放 Bean 工厂,这些工厂能够在需要时生成 Bean 实例。当 Spring 检测到循环依赖时,它会将尚未初始化的 Bean 的工厂放入三级缓存中,并在需要时通过该工厂创建 Bean 实例。

解决过程

当 Spring 容器创建 Bean A 时,如果发现 Bean A 依赖于 Bean B,并且 Bean B 又依赖于 Bean A,Spring 会按照以下步骤处理循环依赖:

  1. 创建 Bean A:首先,Spring 容器会尝试创建 Bean A 的实例。
  2. 检查缓存:在创建过程中,Spring 会检查一级缓存(singletonObjects)和二级缓存(earlySingletonObjects)中是否已经存在 Bean A 的实例。如果不存在,则继续创建。
  3. 放入三级缓存:在 Bean A 的实例化阶段完成后(即调用了构造器),但属性填充和初始化尚未开始时,Spring 会将 Bean A 的工厂放入三级缓存(singletonFactories)中。
  4. 创建 Bean B :接下来,Spring 容器会创建 Bean B 的实例。在创建过程中,如果 Bean B 依赖于 Bean A,Spring 会从三级缓存中获取 Bean A 的工厂,并调用其 getObject 方法来创建 Bean A 的早期引用(即部分初始化的 Bean A 实例),然后将其注入到 Bean B 中。
  5. 完成 Bean B 的创建:Bean B 完成创建后,会被放入一级缓存(singletonObjects)中。
  6. 完成 Bean A 的创建:最后,Spring 容器会完成 Bean A 的属性填充和初始化,并将其放入一级缓存(singletonObjects)中。此时,Bean A 和 Bean B 都已经完成了完整的初始化过程,并且没有发生循环依赖错误。
相关推荐
架构师沉默6 小时前
设计多租户 SaaS 系统,如何做到数据隔离 & 资源配额?
java·后端·架构
Java中文社群8 小时前
重要:Java25正式发布(长期支持版)!
java·后端·面试
沐怡旸9 小时前
【底层机制】std::string 解决的痛点?是什么?怎么实现的?怎么正确用?
c++·面试
每天进步一点_JL9 小时前
JVM 类加载:双亲委派机制
java·后端
用户298698530149 小时前
Java HTML 转 Word 完整指南
java·后端
渣哥9 小时前
原来公平锁和非公平锁差别这么大
java
渣哥10 小时前
99% 的人没搞懂:Semaphore 到底是干啥的?
java
bobz96510 小时前
QoS 中的优先级相关的设计
面试
J2K10 小时前
JDK都25了,你还没用过ZGC?那真得补补课了
java·jvm·后端
kfyty72510 小时前
不依赖第三方,不销毁重建,loveqq 框架如何原生实现动态线程池?
java·架构