Springboot循环依赖

什么是循环依赖

循环依赖(Circular Dependency) 是指两个或多个 Bean 相互直接或间接依赖,导致容器无法正常初始化这些 Bean。

@Service
public class ServiceA {
    @Autowired
    private ServiceB serviceB; // ServiceA 依赖 ServiceB
}

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

Spring Boot 基于 Spring 框架,其循环依赖的处理机制与 Spring 一致,但在 Spring Boot 2.6+ 版本中默认禁止了循环依赖(通过 spring.main.allow-circular-references=false)。 

产生循环依赖的原因

1.构造函数注入循环依赖

@Service
public class ServiceA {
    private final ServiceB serviceB;
    public ServiceA(ServiceB serviceB) { // 构造函数注入
        this.serviceB = serviceB;
    }
}

@Service
public class ServiceB {
    private final ServiceA serviceA;
    public ServiceB(ServiceA serviceA) { // 构造函数注入
        this.serviceA = serviceA;
    }
}
  • 直接报错:构造函数注入的循环依赖无法解决,容器启动时抛出 BeanCurrentlyInCreationException

2.Setter/Field 注入循环依赖

在spring中使用@Autowired注解标签进行自动注入,如果不加以处理,会出现循环依赖问题 。

怎么解决循环依赖

在Springboot2.5以前可以通过三级缓存解决单例 Bean 的循环依赖问题。

缓存名称职责
singletonObjects存放完全初始化好的 Bean(一级缓存)
earlySingletonObjects存放提前暴露的早期 Bean(二级缓存)
singletonFactories存放 Bean 的工厂对象(三级缓存)

以最初的ServiceA与ServiceB为例,

  1. 创建 ServiceA,通过工厂将其半成品引用存入三级缓存。

  2. ServiceA 注入 ServiceB,触发 ServiceB 的创建。

  3. 创建 ServiceB,同样将其半成品引用存入三级缓存。

  4. ServiceB 注入 ServiceA 时,从三级缓存中获取 ServiceA 的早期引用,完成 ServiceB 的初始化。

  5. ServiceB 初始化完成后,ServiceA 完成依赖注入,最终初始化。

 出现循环依赖之后的几个解决思路:

1.避免循环依赖(推荐)

  • 重构代码:将公共逻辑抽离到第三个 Bean 中。

  • 使用接口或抽象类:通过面向接口编程解耦具体实现。

2. 允许循环依赖(临时方案)

在 application.properties 中显式允许循环依赖:

# Spring Boot 2.6+ 需要手动开启
spring.main.allow-circular-references=true

这种只适用于Springboot版本在2.6以上的循环依赖被禁止的情形。 

3. 使用 @Lazy 延迟加载

在其中一个依赖上添加 @Lazy,延迟注入 Bean 的初始化:

@Service
public class ServiceA {
    @Lazy
    @Autowired
    private ServiceB serviceB; // 延迟初始化 ServiceB
}

4. 调整注入方式

 优先使用 Setter/Field 注入:避免构造函数注入导致的不可解循环依赖。

@Service
public class ServiceA {
    private ServiceB serviceB;
    
    @Autowired
    public void setServiceB(ServiceB serviceB) { // Setter 注入
        this.serviceB = serviceB;
    }
}

 使用setter注入

循环依赖的局限性

  • 构造函数注入无法解决循环依赖:Spring 容器在创建 Bean 时需先完成构造函数调用,此时依赖的 Bean 尚未初始化。

  • 原型(Prototype)作用域的 Bean:Spring 不管理原型 Bean 的完整生命周期,无法解决其循环依赖。

  • AOP 代理问题:如果 Bean 被 AOP 代理(如 @Async@Transactional),可能导致循环依赖解决失败。

总结

Spring Boot 的循环依赖本质是 Spring 框架的机制问题,解决核心在于:

  1. 理解三级缓存的工作原理。

  2. 优先通过代码设计避免循环依赖。

  3. 必要时合理使用 @Lazy 或调整注入方式。

 尽可能在设计之初就避免循环依赖

循环依赖是指两个或多个类相互引用对方,导致在初始化bean时出现无法确定先后顺序的情况。在Spring Boot中,循环依赖是不鼓励的,默认情况下是禁止的。当出现循环依赖时,Spring Boot会抛出一个错误,并提供解决方法。 解决循环依赖的方式有两种: 1. 延迟加载:可以通过使用@Lazy注解来延迟加载其中一个bean,从而解决初始化顺序的问题。在需要互相依赖的bean上添加@Lazy注解,使其在真正被使用时才进行初始化。例如,可以在@Autowired注解上添加@Lazy注解: ``` @Autowired @Lazy private TokenService tokenService; ``` 2. 修改配置来打破循环:可以通过修改配置文件来允许循环依赖。在application.yml或application.properties文件中添加配置项spring.main.allow-circular-references: true即可允许循环依赖。请注意,这只是作为最后的手段,不推荐经常使用。 总之,循环依赖是一种不鼓励且默认禁止的情况,在Spring Boot中可以通过延迟加载或修改配置来解决循环依赖的问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [springboot解决循环依赖问题](https://blog.csdn.net/qq_41175917/article/details/125912639)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值