spring是如何解决循环依赖的(二级缓存不行吗)?

首先我们要知道什么是循环依赖,在 Spring 中,循环依赖指的是两个或多个 Bean 之间相互引用(如 A 依赖 B,B 又依赖 A)。看起来像死锁一样是吧。

核心思路:

Spring 通过三级缓存机制解决了单例 Bean 之间的循环依赖问题。通过三级缓存,存储不同的bean示例,通过提前暴露未完全初始化的bean 打破循环,流程如下:

三级缓存的定义:

一级缓存(singletonObjects):

存储完全初始化的bean实例,是最终可直接使用的实例。

二级缓存(earlySingletonObjects

存储提前暴露的未完全初始化的单例bean(已经实例化但是未完成属性注入和初始化方法)

三级缓存(singletonFactories

存储bean的工厂对象,用于需要时生成bean的早起代理对象或者原始实例。

解决循环依赖的步骤(以 A 依赖 B,B 依赖 A 为例)

  1. 初始化 A

    • 调用 getBean(A),检查三级缓存均无 A,开始创建 A。
    • 实例化 A(执行构造方法),但未注入属性和执行初始化方法。
    • 将 A 的工厂对象(ObjectFactory)放入三级缓存(singletonFactories),用于后续生成早期实例。
  2. A 依赖 B,触发 B 的初始化

    • A 需要注入 B,调用 getBean(B),检查三级缓存均无 B,开始创建 B。
    • 实例化 B,将 B 的工厂对象放入三级缓存。
    • B 需要注入 A,调用 getBean(A)
  3. 获取 A 的早期实例

    • 查找 A 时,一级缓存无,但三级缓存有 A 的工厂对象。
    • 通过工厂对象生成 A 的早期实例(若 A 需要代理,此处会生成代理对象),并将其从三级缓存移至二级缓存(earlySingletonObjects)。
    • 将 A 的早期实例注入 B,B 完成属性注入和初始化,放入一级缓存(singletonObjects)。
  4. 完成 A 的初始化

    • B 初始化完成后,将 B 注入 A,A 完成属性注入和初始化,放入一级缓存。
    • 最终,A 和 B 均在一级缓存中,循环依赖解决。

关键细节

  • 仅支持单例 Bean:原型(prototype)Bean 不支持循环依赖,因为原型 Bean 每次获取都会新建,无法提前暴露。
  • 构造器循环依赖无法解决 :若循环依赖发生在构造方法(如 A 的构造器依赖 B,B 的构造器依赖 A),Spring 会直接抛出 BeanCurrentlyInCreationException,因为此时 Bean 尚未实例化,无法提前暴露。
  • 三级缓存的作用:三级缓存通过工厂对象延迟生成早期实例(尤其是代理对象),避免提前生成代理导致的问题(如初始化方法执行时机错误)。
相关推荐
zhanghongbin01几秒前
AI 采集器:Claude Code、OpenAI、LiteLLM 监控
java·前端·人工智能
计算机毕设vx_bysj68693 分钟前
【免费领源码】77196基于java的手机银行app管理系统的设计与实现 计算机毕业设计项目推荐上万套实战教程JAVA,node.js,C++、python、大屏数据可视化
java·mysql·智能手机·课程设计
忘梓.3 分钟前
墨色规则与血色节点:C++红黑树设计与实现探秘
java·开发语言·c++
hhh3u3u3u3 分钟前
Visual C++ 6.0中文版安装包下载教程及win11安装教程
java·c语言·开发语言·c++·python·c#·vc-1
星河耀银海6 分钟前
C++ 模板进阶:特化、萃取与可变参数模板
java·开发语言·c++
格鸰爱童话23 分钟前
向AI学习项目技能(五)
java·学习
程序员萌萌25 分钟前
Java之mysql实战讲解(三):聚簇索引与非聚簇索引
java·mysql·聚簇索引
好家伙VCC40 分钟前
**发散创新:基于Python与ROS的机器人运动控制实战解析**在现代机器人系统开发中,**运动控制**是实现智能行为的核心
java·开发语言·python·机器人
程途知微1 小时前
ConcurrentHashMap线程安全实现原理全解析
java·后端
Mars酱1 小时前
1分钟编写贪吃蛇 | JSnake贪吃蛇单机版
java·后端·开源