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

循环依赖的定义

循环依赖指两个或多个Bean相互依赖,形成闭环。例如Bean A依赖Bean B,而Bean B又依赖Bean A,导致Spring容器无法正常初始化。

使用构造函数注入避免循环依赖

Spring官方推荐优先使用构造函数注入而非字段注入。构造函数注入在启动时就能检测到循环依赖并抛出异常,避免运行时问题。

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

@Service
public class ServiceB {
    private final ServiceA serviceA;
    
    @Autowired
    public ServiceB(ServiceA serviceA) {
        this.serviceA = serviceA;
    }
}

此时启动会直接报错:Requested bean is currently in creation

使用@Lazy延迟加载

在其中一个依赖上添加@Lazy注解,延迟依赖的初始化,打破循环链。

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;
}

@Service
public class ServiceB {
    @Autowired
    private ServiceA serviceA;
}

重新设计代码结构

最佳实践是通过中间层或接口解耦:

  1. 提取公共逻辑到第三方Bean
  2. 使用事件驱动(ApplicationEventPublisher
  3. 采用面向接口编程

调整Bean加载顺序

通过@DependsOn显式指定加载顺序:

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

注意事项

  • Spring默认允许单例Bean的循环依赖,但原型(Prototype)作用域的Bean会直接报错
  • 循环依赖可能暴露未完全初始化的Bean,导致NPE风险
  • 建议通过Sonar等工具检测循环依赖代码坏味道
相关推荐
古城小栈2 分钟前
边缘计算:K3s 轻量级 K8s 部署实践
java·kubernetes·边缘计算
武子康2 分钟前
Java-196 消息队列选型:RabbitMQ vs RocketMQ vs Kafka
java·分布式·kafka·rabbitmq·rocketmq·java-rocketmq·java-rabbitmq
VX:Fegn08953 分钟前
计算机毕业设计|基于springboot + vue超市管理系统(源码+数据库+文档)
数据库·vue.js·spring boot·后端·课程设计
m0_740043736 分钟前
SpringBoot02-SpringMVC入门
java·开发语言·spring boot·spring·mvc
Seven977 分钟前
字符串匹配算法
java
ss2738 分钟前
阻塞队列:生产者-消费者模式
java·开发语言
IT_陈寒8 分钟前
JavaScript 开发者必知的 7 个 ES2023 新特性,第5个能让代码量减少50%
前端·人工智能·后端
艾莉丝努力练剑10 分钟前
【Linux进程(一)】深入理解计算机系统核心:从冯·诺依曼体系结构到操作系统(OS)
java·linux·运维·服务器·git·编辑器·操作系统核心
Kiri霧10 分钟前
Go 字符串格式化
开发语言·后端·golang
guslegend10 分钟前
SpringBoot 缓存深入
java