【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的性能影响
相关推荐
Flittly5 小时前
【AgentScope Java新手村系列】(16)从RAG到多路检索
java·spring boot·spring
小兔崽子去哪了5 小时前
Java 生成二维码解决方案
java·后端
人活一口气9 小时前
从JVM调优到MCP协议:Java全栈技术体系深度总结与企业级架构实践
java·spring boot
NE_STOP11 小时前
Vibe Coding -- 完整项目案例实操
java
荣码11 小时前
GraphRAG:普通RAG只能回答"点"的问题,我踩了4个坑才搞懂
java·python
SimonKing11 小时前
Google第三方授权登录
java·后端·程序员
明月光81811 小时前
从一行 @Builder 说起:重新拾起 Java 的 Lombok、注解与 Builder 模式
java
考虑考虑20 小时前
Mybatis实现批量插入
java·后端·mybatis
咖啡八杯21 小时前
GoF设计模式——中介者模式
java·后端·spring·设计模式
青石路1 天前
记一次多JDK版本问题的排查,一坑套一坑,差点没爬上来
java