什么是循环依赖

当一个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. 将html代码部署到阿里云服务器,并进行域名解析,以及在部署过程中遇到的问题和解决方法

    本博客主要是说一下,,如何将html代码部署到阿里云服务器,并进行域名解析,以及在部署过程中遇到的问题和解决方法. 1.先在阿里云上购买一台阿里云服务器(ECS云服务器): 2.远程连接上该服务器,在 ...

  2. node 连接数据库异常

    1 找不到mysql模块 报错:Cannot find module 'mysql' 处理:npm install mysql 2 建立了多次连接 报错:Cannot enqueue Handshak ...

  3. mysql 随笔

    (select GROUP_CONCAT(car_brand_name separator ',') carBrandName,supplier_id from ycej_supplier_carbr ...

  4. Docker - CentOS 7 安装

    1. 概述 安装 docker markdown 显示有点问题 代码块里的  后面应该跟一个换行, 但是没有跟 这样会导致部分命令直接执行没有反应 2. 环境 os CentOS7 用户 root 3 ...

  5. Hello 2020D(多重集)

    如果有一对时间对在某一场馆有时间重合而这一对时间对在另一场馆没有时间重合,则输出NO,否则输出YES. #define HAVE_STRUCT_TIMESPEC #include<bits/st ...

  6. shell查找七天之前的文件

    #!/bin/bashaweekago=`date -d "7 days ago" +%s`for f in $(ls) do stat -c %Y ${f} aa=`stat - ...

  7. 用Eclipse+Maven+Jetty构建Java Web开发环境(详细笔记)

    (软件环境) 『系统』Windows 10 x64 『JAVA』JDK 1.8.0_91 『Eclipse』 Eclipse-oxygen 『Maven』 apache-maven-3.6.3 『Je ...

  8. ehcache注解全面解析

    通过ehcache以编程方式使用缓存: 跟上面的方式相同,但是缓存通过ehcache去管理,当然比使用map有N多种好处,比如缓存太大了快达到上限之后,将哪一部分缓存清除出去.这种方式完全是通过代码的 ...

  9. selenium Python实现附件上传

    对于web页面的上传功能一般有两类实现方式:一类是将本地文件的路径作为一个值放在input标签中,通过form表单将这个值提交给服务器:另一个类是插件上传,一般基于flash/javascript或者 ...

  10. IIS-详解IIS中URL重写工具的规则条件(Rule conditions)

    出处:https://shiyousan.com/post/635654920639643421 本文结合官方文档和相关示例,详细记录了在IIS中URL重写工具下的规则条件(Rule conditio ...