什么是循环依赖

当一个ClassA依赖于ClassB,然后ClassB又反过来依赖ClassA,这就形成了一个循环依赖:

ClassA -> ClassB -> ClassA

原创声明

本文发布于掘金号【Happyjava】。Happy的掘金地址:https://juejin.im/user/5cc2895df265da03a630ddca,Happy的个人博客:(http://blog.happyjava.cn)[http://blog.happyjava.cn]。欢迎转载,但须保留此段声明。

Spring的循环依赖问题

当你使用构造注入依赖的时候,就有可能发生循环依赖然后报错的问题。什么是构造注入呢?可以看如下代码:

假设有ClassA和ClassB如下:

ClassA.java

@Data
@Component
public class ClassA { private final ClassB classB; public ClassA(ClassB classB) {
this.classB = classB;
}
}

ClassB.java

@Data
@Component
public class ClassB { private final ClassA classA; public ClassB(ClassA classA) {
this.classA = classA;
}
}

就是在类的构造方法里,把依赖注入,这就是所说的构造注入。

构造注入,也是Spring团队推荐的Spring依赖注入的方式(依赖来自IDEA的提示):

虽然是Spring的官方推荐,但是这种方式就是容易出现循环依赖导致程序跑不起来的情况:

当然,也存在多种解决循环依赖的办法,下面一一演示。

重新设计代码

当出现循环依赖的时候,可以考虑重新设计下代码。一般来说,当循环依赖问题出现的时候,往往其原因是设计上分层没有处理好,各个类的耦合度高,各自的职责不够单一。

当然,很多时候,我们也没有那么多时间去重新设计代码。那么,我们可以采取别的方式。

使用懒加载

可以通过Spring提供的@Lazy注解,让Spring懒加载,即当真正需要使用到该bean的时候,再去加载。如,我给上面的示例代码的ClassB的构造方法加入@Lazy注解:

@Data
@Component
public class ClassB { private final ClassA classA; public ClassB(@Lazy ClassA classA) {
this.classA = classA;
}
}

再次启动,就会发现循环依赖报错问题不存在了。

直接使用Autowired单独注入

直接使用@Autowired注入依赖,不要使用构造器的方式注入

@Data
@Component
public class ClassB { @Autowired
private ClassA classA; }
@Data
@Component
public class ClassA { @Autowired
private ClassB classB; }

这种方式,也可以解决Spring循环依赖的问题。

使用Setter注入

除了以上两种方式,还可以通过setter的方式来注入依赖。如下:

ClassA.class

@Data
@Component
public class ClassA { private ClassB classB; @Autowired
public void setClassB(ClassB classB) {
this.classB = classB;
}
}

ClassB.class

@Data
@Component
public class ClassB { private ClassA classA; @Autowired
public void setClassA(ClassA classA) {
this.classA = classA;
}
}

通过Setter注入依赖的方式,一样可以解决Spring循环依赖的问题。

总结

使用Spring作为开发框架,一不小心就会碰到循环依赖,程序启动不了的问题。如果真的出现了循环依赖的问题,可以尝试采用上面的几种方式解决。当然,解决的办法还有很多,比如,还可以通过PostConstruct注解来解决(摘抄baeldung的博客):

方法可能有很多种,就不太深入探讨了。

关注公众号领资料

搜索公众号【Happyjava】,回复【电子书】和【视频】,即可获取大量优质电子书和大数据、kafka、nginx、MySQL等视频资料

Spring中解决循环依赖报错的问题的更多相关文章

  1. Spring:解决因@Async引起的循环依赖报错

    最近项目中使用@Async注解在方法上引起了循环依赖报错: org.springframework.beans.factory.BeanCurrentlyInCreationException: Er ...

  2. 【Spring】Spring中的循环依赖及解决

    什么是循环依赖? 就是A对象依赖了B对象,B对象依赖了A对象. 比如: // A依赖了B class A{ public B b; } // B依赖了A class B{ public A a; } ...

  3. 彻底理解Spring如何解决循环依赖

    Spring bean生命周期 可以简化为以下5步. 1.构建BeanDefinition 2.实例化 Instantiation 3.属性赋值 Populate 4.初始化 Initializati ...

  4. Spirng 循环依赖报错:Requested bean is currently in creation: Is there an unresolvable circular reference?

    1:前言 最近在项目中遇到了一次循环依赖报错的问题,虽然解决的很快,但是有些不明白的地方,特此记录. 在此我把 bean 的结构和 注入方式单独拎出来进行演示 1.1:报错提示 1.2:错误日志 Ex ...

  5. 从一部电影史上的趣事了解 Spring 中的循环依赖问题

    title: 从一部电影史上的趣事了解 Spring 中的循环依赖问题 date: 2021-03-10 updated: 2021-03-10 categories: Spring tags: Sp ...

  6. 面试必杀技,讲一讲Spring中的循环依赖

    本系列文章: 听说你还没学Spring就被源码编译劝退了?30+张图带你玩转Spring编译 读源码,我们可以从第一行读起 你知道Spring是怎么解析配置类的吗? 配置类为什么要添加@Configu ...

  7. 面试阿里,腾讯,字节跳动90%都会被问到的Spring中的循环依赖

    前言 Spring中的循环依赖一直是Spring中一个很重要的话题,一方面是因为源码中为了解决循环依赖做了很多处理,另外一方面是因为面试的时候,如果问到Spring中比较高阶的问题,那么循环依赖必定逃 ...

  8. Spring 如何解决循环依赖问题?

    在关于Spring的面试中,我们经常会被问到一个问题,就是Spring是如何解决循环依赖的问题的. 这个问题算是关于Spring的一个高频面试题,因为如果不刻意研读,相信即使读过源码,面试者也不一定能 ...

  9. Spring如何解决循环依赖问题

    目录 1. 什么是循环依赖? 2. 怎么检测是否存在循环依赖 3. Spring怎么解决循环依赖 本文主要是分析Spring bean的循环依赖,以及Spring的解决方式. 通过这种解决方式,我们可 ...

随机推荐

  1. [Reversal 剧情设计] 序言

    想开发一个图形游戏已经很久了,这个寒假好不容易学了一些OpenGL知识,初步具备了开发图形游戏的能力,我觉得我可以开始了.当然,我的游戏并不是冷冰冰的PVE,肯定还需要一些剧情丰富色彩. 本来我的游戏 ...

  2. JDBC 基础用法学习

    JDBC概述 java 数据库链接,sun公司退出的 java 访问数据库的标准规范接口 是一种用于执行SQL语句的 java API 可以作为多种关系数据库提供统一接口 是一组 java 工具类和接 ...

  3. Fiddler修改http请求响应简单实例

    Fiddler是一个http调试代理,它能够记录并检查所有你的电脑和互联网之间的http通讯. 主要功能 设置断点,查看Fiddle说有的进出的数据(指cookie,html,js,css等文件,这些 ...

  4. RocketMq-粪发涂墙1.0

    角色 说明 Producer 生产者,用于将消息发送到RocketMQ,生产者本身既可以是生成消息,也可以对外提供接口,由外部来调用接口,再由生产者将受到的消息发送给MQ. Consumer 消费者, ...

  5. WPscan扫描工具安装使用

    WPScan是Kali Linux默认自带的一款漏洞扫描工具,它采用Ruby编写,能够扫描WordPress网站中的多种安全漏洞,其中包括WordPress本身的漏洞.插件漏洞和主题漏洞.最新版本WP ...

  6. tomcat服务器启动执行的两个方法

    第一 SetApplicationContext(需要继承ApplicationContextAware)重写 第二 ContextInitialize(需要继承servleContet)重写,(co ...

  7. Python - 运行流程图, call graph, 调用图

    解决方案 pycallgraph(感觉直接用pycallgraph grahviz命令生成的图并不是我想要的) 如何去阅读并学习一些优秀的开源框架的源码? - mailto1587的回答 - 知乎 h ...

  8. Dirjkstra

    Description 给定n个点,m条有向边 求每个点到1号点的最短距离 Input 第一行两个数为n,m,n表示顶点个数,m表示边的条数. (1 ≤ n, m ≤ 100 ) 接下来m行,每一行有 ...

  9. Could not transfer artifact org.springframework.boot:spring-boot-starter-parent:pom:2.1.9.RELEASE from/to 阿里云镜像地址

    今天从 http://start.spring.io/ 下载的demo项目,导入eclipse后,pom文件一直报 parent包错,然后感觉就是自己maven镜像里面搜不到这个包, 所以改了 mav ...

  10. [数据库] MariaDB安装及使用

    一.安装MariaDB 1.使用官方源安装marisdb 如果使用阿里云的源,目前的版本号为5.5.64.如果想安装最新的10.x版本,则需要使用MariaDB的官方源. 1)配置官方源: 在/etc ...