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 都已经完成了完整的初始化过程,并且没有发生循环依赖错误。
相关推荐
心之伊始4 小时前
Java 后端接入大模型:从 Token、并发到推理成本的完整估算方法
java·spring boot·性能优化·大模型·llm
BlackTurn5 小时前
技术经理投标
java
YG亲测源码屋5 小时前
java配置环境变量、jdk环境变量配置、java环境变量设置方法
java·开发语言
MIUMIUKK5 小时前
从语法层面,看懂 Python 的特殊处
java·开发语言·python
hujinyuan201605 小时前
2026年3月 中国电子学会青少年软件编程(Python)三级考试试卷 真题及答案
java·python·算法
basketball6166 小时前
C++ 高级编程:2. 基本线程池实现
java·开发语言·c++
MageGojo6 小时前
天气 API 接入实战:基于 ApiZero 实现实时天气、分钟级降水和 15 天预报查询
java·后端·spring·api 接口接入·接口实战
代码小库6 小时前
免费制作简历 + 免费简历押题
面试
自动跟随6 小时前
UWB自动跟随技术全栈解析:从定位算法到“位控一体化“
java·网络·人工智能
Aphasia3116 小时前
手写KeepAlive组件
前端·react.js·面试