当前位置: 首页 > news >正文

Spring 怎么解决循环依赖问题?

Spring 循环依赖(circular dependency) 指的是多个 Bean 之间的相互依赖,比如:

  • A 依赖 B,B 又依赖 A;
  • 或者 A → B → C → A 这种嵌套循环依赖。

这是一个常见又棘手的问题,但 Spring 是可以解决部分类型的循环依赖的! 🌱
下面我们来详细说说 Spring 是 怎么解决循环依赖的,以及在哪些情况下解决不了

🔁 一、Spring 解决循环依赖的核心机制:三级缓存

Spring 的 Bean 生命周期中,在创建单例 Bean 时,使用了三级缓存机制来应对循环依赖问题。

✅ Spring 可解决的循环依赖:

构造方法之外的依赖注入(也称为Setter 注入 / 字段注入

📦 三级缓存的具体解释

Spring 为了避免无限递归创建 Bean,使用了 3 个缓存(都在 DefaultSingletonBeanRegistry 中):

缓存名类型作用
singletonObjects一级缓存存放完全初始化好的单例 Bean(可直接使用)
earlySingletonObjects二级缓存存放还未初始化完成的早期 Bean(已实例化但还没注入依赖)
singletonFactories三级缓存存放 ObjectFactory(可生成早期 Bean 的代理对象)

🧩 举个简单例子:A → B → A

@Component
public class A {
    @Autowired
    private B b;
}

@Component
public class B {
    @Autowired
    private A a;
}

Spring 的处理过程如下:

  1. 创建 A(A 被标记为 “正在创建”)
  2. A 依赖 B,开始创建 B
  3. B 又依赖 A,发现 A 正在创建 → 从三级缓存中拿 A 的早期引用(代理或实例)
  4. B 注入了 A 的早期引用,B 创建完成
  5. 回到 A,注入 B,A 创建完成
  6. 把 A 的完整对象放入一级缓存

✅ 最终,A 和 B 都创建成功,并互相注入。

🚫 二、Spring 无法解决的循环依赖

1. 构造器注入导致的循环依赖(Constructor Injection)

@Component
public class A {
    public A(B b) {}
}

@Component
public class B {
    public B(A a) {}
}
  • 构造函数中就需要对方的完整对象
  • 此时 Spring 无法提前暴露早期对象
  • 所以抛出:BeanCurrentlyInCreationException

解决方法:

  • 将构造器注入改为字段或 setter 注入

2. 原型(@Scope("prototype"))Bean 的循环依赖

  • Spring 只对 单例 Bean 做了三级缓存处理
  • 原型 Bean 每次都新建,Spring 不会缓存它
  • 所以 原型作用域的循环依赖,Spring 无法解决

3. 通过 AOP 代理增强的 Bean,未提前暴露代理对象

有些 AOP 场景中,Spring 无法及时将代理对象放入三级缓存,也会导致依赖注入失败。

✅ 三、开发中如何避免/解决循环依赖

✅ 建议一:优先使用构造器注入 + 明确设计依赖关系

  • 构造器注入更安全、清晰,但要避免循环
  • 如果存在循环依赖,说明你的设计可能需要重构

✅ 建议二:使用 @Lazy 延迟注入

  • Spring 会在需要时再去注入依赖,打破初始化顺序
@Autowired
@Lazy
private A a;

✅ 建议三:提取公共依赖,重构服务划分

  • 将 A 和 B 共同依赖的部分抽成一个新的类 C

🧠 总结:Spring 怎么解决循环依赖?

类型是否能解决原因
Setter/字段注入 + 单例 Bean✅ 可以通过三级缓存(提前暴露早期 Bean 引用)
构造器注入❌ 不行无法在构造阶段暴露代理对象
原型 Bean❌ 不行Spring 不缓存原型 Bean
AOP复杂代理⚠️ 有条件支持看 Spring 是否能生成早期代理对象

相关文章:

  • 室内指路机器人是否支持与第三方软件对接?
  • 2025年渗透测试面试题总结-某四字大厂面试复盘扩展 二面 (题目+回答)
  • JavaScript 中的 Reflect 详解
  • notepad++8.6.4安装及细节
  • 【Python Cookbook】数字日期和时间(一)
  • unity的dots中instantiate克隆对象后,对象会在原位置闪现的原因和解决
  • 吉卜力动画风格图像生成:Ghibli Diffusion
  • SDL中SDL_AudioSpec结构体参数
  • 23 MVC模式
  • Android学习总结之网络篇(HTTP请求流程)
  • Day82 | 灵神 | 快慢指针 重排链表
  • 科技赋能消防:无人机“挂弹灭火“构筑森林防火墙!
  • 生成式人工智能认证(GAI认证)如何推动就业市场的创新?
  • 【橘子大模型】关于记忆上下文Message History
  • Jenkins学习(B站教程)
  • [从零开始学习JAVA] 初学网络编程
  • 设计模式简述(十二)策略模式
  • [GN] sigrokdecode 模块学习指南 --- 准备阶段
  • COZE通关指南:工作流与插件开发
  • Go语言-初学者日记(四):包管理
  • wordpress主题演示站点/优化推荐
  • 网站乱码解决办法/专门发广告的app
  • 如何用网站开发工具停止网页进程/网络营销策略概念
  • 中国人去菲律宾做网站赌钱会抓吗/中国seo高手排行榜
  • 淘宝有做钓鱼网站的吗/百度官网地址
  • 注册域名的服务商平台/百度seo关键词优化公司