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 都已经完成了完整的初始化过程,并且没有发生循环依赖错误。
相关推荐
似水明俊德17 小时前
02-C#.Net-反射-面试题
开发语言·面试·职场和发展·c#·.net
Leinwin17 小时前
OpenClaw 多 Agent 协作框架的并发限制与企业化规避方案痛点直击
java·运维·数据库
薛定谔的悦17 小时前
MQTT通信协议业务层实现的完整开发流程
java·后端·mqtt·struts
enjoy嚣士18 小时前
springboot之Exel工具类
java·spring boot·后端·easyexcel·excel工具类
罗超驿18 小时前
独立实现双向链表_LinkedList
java·数据结构·链表·linkedlist
无限大618 小时前
AI实战03:Java开发岗专属工作流|用AI辅助代码审查与文档生成
面试
盐水冰19 小时前
【烘焙坊项目】后端搭建(12) - 订单状态定时处理,来单提醒和顾客催单
java·后端·学习
凸头19 小时前
CompletableFuture 与 Future 对比与实战示例
java·开发语言
wuqingshun31415919 小时前
线程安全需要保证几个基本特征
java·开发语言·jvm
程序员雨果19 小时前
软件测试工程师:面试题与经验分享
软件测试·面试·职场和发展