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

Spring Boot循环依赖的陷阱与解决方案:如何打破“Bean创建死循环”?

引言

在Spring Boot开发中,你是否遇到过这样的错误信息?
The dependencies of some of the beans in the application context form a cycle
这表示你的应用出现了循环依赖。尽管Spring框架通过巧妙的机制解决了部分循环依赖问题,但在实际开发中(尤其是使用构造器注入时),开发者仍需警惕此类问题。本文将深入探讨循环依赖的根源,分析Spring的解决策略,并提供多种实战解决方案。


一、什么是循环依赖?

循环依赖指两个或多个Bean相互依赖对方,形成一个闭环。例如:

  • Bean A​ 的创建需要注入 ​Bean B
  • Bean B​ 的创建又需要注入 ​Bean A

此时,Spring容器在初始化Bean时会陷入“死循环”。以下是一个典型示例:

@Service
public class ServiceA {private final ServiceB serviceB;public ServiceA(ServiceB serviceB) { // 构造器注入ServiceBthis.serviceB = serviceB;}
}@Service
public class ServiceB {private final ServiceA serviceA;public ServiceB(ServiceA serviceA) { // 构造器注入ServiceAthis.serviceA = serviceA;}
}

启动应用时,Spring会抛出异常:
BeanCurrentlyInCreationException: Error creating bean with name 'serviceA': Requested bean is currently in creation


二、Spring如何解决循环依赖?

Spring通过三级缓存机制解决单例Bean的循环依赖问题:

  1. 一级缓存​(singletonObjects):存放完全初始化好的Bean。
  2. 二级缓存​(earlySingletonObjects):存放提前曝光的半成品Bean(仅实例化,未填充属性)。
  3. 三级缓存​(singletonFactories):存放Bean的工厂对象,用于生成半成品Bean。

解决流程​(以A和B相互依赖为例):

  1. 创建A时,先实例化A(未填充属性),并将A的工厂放入三级缓存。
  2. 填充A的属性时发现需要B,开始创建B。
  3. 创建B时,实例化B后,发现需要A,此时从三级缓存中通过工厂获取A的半成品对象。
  4. B完成初始化,放入一级缓存。
  5. A继续填充B的实例,完成初始化,放入一级缓存。

关键限制​:该机制仅支持单例Bean通过属性注入的场景。​构造器注入会直接失败!


三、为何构造器注入会导致循环依赖失败?

构造器注入要求Bean在实例化阶段立即获得依赖对象,而三级缓存机制需要在属性注入阶段解决依赖。因此,当两个Bean都使用构造器注入时,Spring无法提前曝光半成品Bean,导致循环依赖无法解决。


四、解决方案:打破循环依赖的四种方法
1. ​改用Setter/Field注入(谨慎使用)​

将构造器注入改为Setter或字段注入,允许Spring延迟注入依赖:

@Service
public class ServiceA {private ServiceB serviceB;@Autowired // Setter注入public void setServiceB(ServiceB serviceB) {this.serviceB = serviceB;}
}
  • 优点​:快速解决问题。
  • 缺点​:破坏了不可变性(字段非final),且可能掩盖设计问题。
2. ​使用@Lazy延迟加载

在依赖对象上添加@Lazy,告知Spring延迟初始化Bean:

@Service
public class ServiceA {private final ServiceB serviceB;public ServiceA(@Lazy ServiceB serviceB) {this.serviceB = serviceB; // 实际注入的是代理对象}
}
  • 原理​:Spring生成代理对象,只有在首次调用时才会真正初始化目标Bean。
  • 适用场景​:解决构造函数注入的循环依赖。
3. ​重新设计代码结构

通过分层或提取公共逻辑,消除循环依赖:

  • 方案一​:引入中间层(如ServiceC),将A和B的共同依赖转移到C。
  • 方案二​:使用事件驱动(ApplicationEvent),解耦直接依赖。
// 事件驱动示例
@Service
public class ServiceA {@Autowiredprivate ApplicationEventPublisher eventPublisher;public void doSomething() {eventPublisher.publishEvent(new EventA());}
}@Service
public class ServiceB {@EventListenerpublic void handleEventA(EventA event) {// 处理事件}
}
4. ​使用ObjectProvider(推荐)​

在构造器中注入ObjectProvider,按需获取依赖:

@Service
public class ServiceA {private final ServiceB serviceB;public ServiceA(ObjectProvider<ServiceB> serviceBProvider) {this.serviceB = serviceBProvider.getIfUnique();}
}
  • 优点​:保持构造器注入的不可变性,显式控制依赖获取时机。
  • 注意​:需确保依赖Bean存在且唯一。

五、最佳实践与预防措施
  1. 优先使用构造器注入​:保持Bean的不可变性和明确依赖,但需警惕循环依赖。
  2. 定期检测循环依赖​:
    • 使用IDE插件(如IntelliJ的Circular Dependencies分析)。
    • 通过Maven/Gradle插件(如spring-boot-dependencies-analysis)。
  3. 代码分层规范​:
    • 严格遵循分层架构(Controller → Service → Repository)。
    • 避免同一层内的Bean相互依赖。
  4. 单元测试验证​:编写集成测试,验证Bean的初始化过程。
@SpringBootTest
public class CircularDependencyTest {@Autowiredprivate ApplicationContext context;@Testvoid contextLoads() {// 若启动无异常,则通过测试assertNotNull(context.getBean(ServiceA.class));}
}

六、总结

循环依赖是Spring开发中的常见陷阱,其本质是代码设计问题。尽管Spring提供了部分解决方案,但重构代码消除循环依赖才是根本之道。通过合理使用注入方式、代码分层和工具检测,开发者可以有效避免此类问题,构建高可维护性的应用。

记住​:

  • 慎用@Lazy和Setter注入,它们可能掩盖设计缺陷。
  • 构造器注入 + 合理分层 = 更健壮的系统!
http://www.xdnf.cn/news/6507.html

相关文章:

  • (面试)Android各版本新特性
  • Oracle学习日记--Oracle中使用单个inert语句实现插入多行记录
  • 支付宝小程序关键词排名优化中的常见错误
  • Linux下载与安装
  • leetcode:58. 最后一个单词的长度(python3解法)
  • SearchClassUtil
  • 102. 二叉树的层序遍历
  • “光伏+储能+智能调控”,CET中电技术分布式智能微网方案如何实现?
  • 多线程(四)
  • 云服务器的运用自如
  • 数学复习笔记 14
  • [CSS3]属性增强1
  • 回调函数应用示例
  • 网络安全-等级保护(等保) 2-5-1 GB/T 25070—2019 附录B (资料性附录)第三级系统安全保护环境设计示例
  • IEC 60601-2-16:2025 标准解析
  • python打卡day27
  • TCP/IP 知识体系
  • 国标GB/T 12536-90滑行试验全解析:纯电动轻卡行驶阻力模型参数精准标定
  • 【AI大模型学习路线】第二阶段之RAG基础与架构——第七章(【项目实战】基于RAG的PDF文档助手)query搜索与文档排序?
  • win10-django项目与mysql的基本增删改查
  • 从代码学习深度学习 - 实战Kaggle比赛:狗的品种识别(ImageNet Dogs)PyTorch版
  • 关于nginx浏览器访问.php直接被当做文件下载相关问题
  • Github 2025-05-16 Java开源项目日报 Top9
  • OM和SCADA的区别
  • 目标检测指标计算
  • C++ I/O多路复用
  • uniapp自定义日历计划写法(vue2)
  • 生信分析进阶15 - 从GTF文件提取起始密码子、终止密码子、外显子剪切供体和受体
  • 基于大模型的脑出血智能诊疗与康复技术方案
  • 计算机组成原理——数据的表示