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 都已经完成了完整的初始化过程,并且没有发生循环依赖错误。
相关推荐
_小马快跑_40 分钟前
Java 的 8 大基本数据类型:为何是不可或缺的设计?
java
想用offer打牌1 小时前
高并发下如何保证接口的幂等性
后端·面试·状态机
Re_zero3 小时前
线上日志被清空?这段仅10行的 IO 代码里竟然藏着3个毒瘤
java·后端
洋洋技术笔记3 小时前
Spring Boot条件注解详解
java·spring boot
牛奶14 小时前
Vue 基础理论 & API 使用
前端·vue.js·面试
牛奶14 小时前
Vue 底层原理 & 新特性
前端·vue.js·面试
NAGNIP15 小时前
一文搞懂深度学习中的通用逼近定理!
人工智能·算法·面试
青青家的小灰灰20 小时前
深入理解事件循环:异步编程的基石
前端·javascript·面试
程序员清风21 小时前
程序员兼职必看:靠谱软件外包平台挑选指南与避坑清单!
java·后端·面试
皮皮林5511 天前
利用闲置 Mac 从零部署 OpenClaw 教程 !
java