【Spring Boot】Spring Boot解决循环依赖

循环依赖的概念

循环依赖是指两个或多个Bean相互依赖,形成闭环。例如Bean A依赖Bean B,Bean B又依赖Bean A。Spring默认情况下会抛出BeanCurrentlyInCreationException异常。

解决方法

使用构造器注入 + @Lazy注解

在构造器注入的场景下,可以通过@Lazy延迟加载其中一个Bean来打破循环:

java 复制代码
@Service
public class ServiceA {
    private final ServiceB serviceB;
    
    @Autowired
    public ServiceA(@Lazy ServiceB serviceB) {
        this.serviceB = serviceB;
    }
}

改用Setter/Field注入

Spring官方推荐使用构造器注入,但Setter/Field注入能自动处理循环依赖:

java 复制代码
@Service
public class ServiceA {
    @Autowired
    private ServiceB serviceB;
}

调整代码结构

最彻底的解决方案是重构代码,提取公共逻辑到第三个Bean中,或使用接口分离依赖关系。

Spring处理循环依赖的机制

Spring通过三级缓存解决Setter/Field注入的循环依赖:

  • 一级缓存:存放完整初始化的Bean
  • 二级缓存:存放早期暴露的原始Bean
  • 三级缓存:存放Bean工厂,用于生成原始Bean

注意事项

  • 构造器注入的循环依赖必须显式解决
  • 循环依赖可能掩盖设计问题,应优先考虑重构
  • 高版本Spring Boot默认禁止循环依赖,可通过配置允许:
properties 复制代码
spring.main.allow-circular-references=true

推荐做法

对于必须存在的循环依赖,建议:

  1. 使用@Lazy注解
  2. 保持最少量的循环关系
  3. 添加文档说明循环的必要性
  4. 监控循环依赖Bean的性能影响
相关推荐
z***89714 小时前
SpringBoot Maven 项目 pom 中的 plugin 插件用法整理
spring boot·后端·maven
憧憬blog4 小时前
【Kiro开发集训营】拒绝“屎山”堆积:在 Kiro 中重构“需求-代码”的血缘关系
java·开发语言·kiro
e***74955 小时前
Spring Security 官网文档学习
java·学习·spring
n***i955 小时前
Java NIO文件操作
java·开发语言·nio
笃行客从不躺平6 小时前
接口幂等性(Idempotency)
java
Hero | 柒7 小时前
JAVA反射机制
java·spring·反射
j***63087 小时前
Springboot项目中线程池使用整理
java·spring boot·后端
likuolei7 小时前
Eclipse 创建 Java 接口
java·数据库·eclipse
q***54757 小时前
Spring Boot 经典九设计模式全览
java·spring boot·设计模式
w***15317 小时前
Spring boot启动原理及相关组件
数据库·spring boot·后端