如何解决SpringBoot2.6及之后版本取消了循环依赖的支持问题

 更新时间:2024年10月29日 11:10:59   作者:天空中那座城  
循环依赖指的是两个或者多个bean之间相互依赖,形成一个闭环,SpringBoot从2.6.0开始默认不允许出现Bean循环引用,解决方案包括在全局配置文件设置允许循环引用存在、在SpringApplicationBuilder添加设置允许循环引用、构造器注入

1、问题

循环依赖指的是两个或者多个bean之间相互依赖,形成一个闭环。直接表现为两个service层互相调用对方。

此时会遇到以下问题:

2、报错

当启动项目时,可能出现程序不能启动的情况,查看调试日志,会提示:

The dependencies of some of the beans in the application context form a cycle...

如下图所示:

根据上述代码片段,应用程序存在以下循环依赖:

1. adminBorrowInfoController 依赖 borrowInfoServiceImpl

2. borrowInfoServiceImpl 依赖 lendServiceImpl

3. lendServiceImpl 依赖 lendItemServiceImpl

4. lendItemServiceImpl 又反过来依赖 lendServiceImpl

这样就形成了一个循环依赖的场景。

原因是SpringBoot 从 2.6.0 开始默认不允许出现 Bean 循环引用。而且这个是在Bean 定义上也就是类上就不允许出现循环引用。

3、解决方案

第1种、在全局配置文件设置允许循环引用存在

升级到Spring Boot 2.7及以上版本,可以通过spring.main.allow-circular-references=true配置属性明确开启循环依赖支持。

但我用的2.6.11也可以哦,2.6以上的可以试试。

spring:
  main:
    allow-circular-references:true

第2种、在SpringApplicationBuilder 添加设置允许循环引用

使用SpringApplicationBuilder来启动Spring Boot应用,并通过allowCircularReferences(true)方法开启了循环依赖支持。

public static void main(String[] args) {
  new SpringApplicationBuilder(DemoApplication.class).allowCircularReferences(true).run(args);
}

第N种、还有很多种供大家了解使用

1. 构造器注入

  • 在类中定义构造器,添加需要依赖的类作为参数
  • 使用@Autowired注解构造器
  • Spring会先实例化依赖类,然后通过构造器注入

2. @Lazy

  • 在导致循环依赖的Bean上添加@Lazy注解
  • Spring会延迟初始化这些Bean,先完成非Lazy的Bean初始化
  • 然后再通过setter注入完成Lazy Bean的初始化

3. ObjectFactory

  • 定义ObjectFactory属性,类型为对应类的ObjectFactory
  • Spring会代理注入ObjectFactory,获取对象时才初始化目标Bean

4. 服务定位器

  • 定义一个统一的服务定位器类
  • Bean直接从定位器获取依赖对象,而不是注入依赖

5. 合并类

  • 将互相依赖的类合并为一个类,避免相互依赖

6. 事件回调

  • 使用事件或者回调方式实现解耦
  • 一个类通过事件通知另一个类执行操作,而不是直接调用

7. 接口编程

  • 类依赖接口,不依赖具体实现
  • 具体实现通过setter注入接口

以上是一些主要的具体实现步骤,可以根据实际情况选择适合的方案。

总体上,出现循环依赖通常意味着系统设计需要优化和解耦,需要重新梳理服务的职责和依赖关系,减少不必要的互相依赖,以提高内聚性和可维护性。

这些仅为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • mybatis自动生成时如何设置不生成Example类详解

    mybatis自动生成时如何设置不生成Example类详解

    这篇文章主要给大家介绍了关于mybatis自动生成时如何设置不生成Example类的相关资料,文中介绍的非常详细,对大家具有一定的参考学习价值,需要的朋友们下面来一起看看吧。
    2017-05-05
  • java利用phantomjs进行截图实例教程

    java利用phantomjs进行截图实例教程

    PlantomJs是一个基于javascript的webkit内核无头浏览器 也就是没有显示界面的浏览器,你可以在基于 webkit 浏览器做的事情,它都能做到。下面这篇文章主要给大家介绍了关于java利用phantomjs进行截图的相关资料,需要的朋友可以参考下
    2018-10-10
  • mybatis plus使用redis作为二级缓存的方法

    mybatis plus使用redis作为二级缓存的方法

    这篇文章主要介绍了mybatis plus使用redis作为二级缓存的方法,本文给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2019-09-09
  • java的json解析类库使用示例

    java的json解析类库使用示例

    这篇文章主要介绍了java的json解析类库使用方法,这里使用Zson解析json,这是一个开源的json处理类库
    2014-03-03
  • SpringBoot一个接口多个实现类的调用方式总结

    SpringBoot一个接口多个实现类的调用方式总结

    这篇文章主要介绍了SpringBoot一个接口多个实现类的调用方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2024-01-01
  • springmvc接收json串,转换为实体类List方法

    springmvc接收json串,转换为实体类List方法

    今天小编就为大家分享一篇springmvc接收json串,转换为实体类List方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-08-08
  • Java的AQS基本原理详细分析

    Java的AQS基本原理详细分析

    这篇文章主要介绍了Java的AQS基本原理详细分析,AQS是Abstract Queued Synchronizer的简称,AQS提供了一种实现阻塞锁和一系列依赖FIFO等待队列的同步器的框架,本文主要讲解分析其基本原理,需要的朋友可以参考下
    2024-01-01
  • Spring基于注解配置AOP详解

    Spring基于注解配置AOP详解

    这篇文章主要介绍了Spring基于注解配置AOP详解,Spring 的 AOP 功能是基于 AspectJ 实现的,支持使用注解声明式定义 AOP 切面,Spring 基于注解配置 AOP 需要启用 AspectJ 自动代理功能,需要的朋友可以参考下
    2023-09-09
  • Spring Boot配置AOP打印日志的全过程

    Spring Boot配置AOP打印日志的全过程

    这篇文章主要给大家介绍了关于Spring Boot配置AOP打印日志的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Spring Boot具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-08-08
  • 命令行使用支持断点续传的java多线程下载器

    命令行使用支持断点续传的java多线程下载器

    java命令行下载器,支持断点续传下载,多线程下载,需要的朋友可以参考下
    2014-02-02

最新评论