nested exception is org.springframework.beans.factory.BeanCurrentlyInCreationException(Spring循环依赖问题)

简介: nested exception is org.springframework.beans.factory.BeanCurrentlyInCreationException(Spring循环依赖问题)

1:问题

最近启动项目时候,遇到如下报错

nested exception is org.springframework.beans.factory.BeanCurrentlyInCreationException: Error creating bean with name 'stockReceiptManager': Bean with name 'stockReceiptManager' has been injected


the bean. This is often the result of over-eager type matching - consider using 'getBeanNamesOfType' with the 'allowEagerInit' flag turned off, for example.

 

2:分析原因

后来在网上找了半天说是依赖循环,检查了一下代码,确实存在循环依赖的现象

spring默认是支持循环依赖,allowCircularRefrence默认为true。也可以手动设置applicationContext.setAllowCircularReferences(false); 产生这个问题原因:in its raw version as part of a circular reference, but has eventually been wrapped.应该是项目中对某个类的方法开启异步@Async导致

 

3:解释循环依赖

首先说一下什么是依赖循环,比如:我现在有一个ServiceA需要调用ServiceB的方法,那么ServiceA就依赖于ServiceB,那在ServiceB中再调用ServiceA的方法,就形成了循环依赖。Spring在初始化bean的时候就不知道先初始化哪个bean就会报错。


由于类A通过构造注入需要类B的实例,而类B通过构造注入需要类B的实例,二者之间相互注入,导致循环引用抛出异常,这是一个典型的先有鸡还是先有蛋的故事。所以在使用spring配置类的循环依赖的关系时,应当尽量避免使用构造注入,而是使用setter注入。

public class ClassA {
    @Autowired
    ClassB classB;
}
 
public class ClassB {
    @Autowired
    ClassA classA ;
} 

那如何解决循环依赖,当然最好的方法是重构你的代码,进行解耦,但是重构不是一时的事情,那就使用下面的方法:

方法一:

<bean id="ServiceDependent1" class="org.xyz.ServiceDependent1" lazy-init="true">
      <constructor-arg ref="Service"/>
</bean>
    
<bean id="ServiceDependent2" class="org.xyz.ServiceDependent2" lazy-init="true">
      <constructor-arg ref="Service"/>
</bean>

方法二:

在你的配置文件中,在互相依赖的两个bean的任意一个加上lazy-init属性。

在你注入bean时,在互相依赖的两个bean上加上@Lazy注解也可以。

以上两种方法都能延迟互相依赖的其中一个bean的加载,从而解决循环依赖的问题。

    @Autowired
    @Lazy
    private ClassA classA;
 
    @Autowired
    @Lazy
    private ClassB classB;

相关文章
|
3月前
|
人工智能 Java Spring
Spring Boot循环依赖的症状和解决方案
Spring Boot循环依赖的症状和解决方案
|
6天前
|
存储 缓存 Java
面试问Spring循环依赖?今天通过代码调试让你记住
该文章讨论了Spring框架中循环依赖的概念,并通过代码示例帮助读者理解这一概念。
面试问Spring循环依赖?今天通过代码调试让你记住
|
3天前
|
缓存 Java Spring
spring如何解决循环依赖
Spring框架处理循环依赖分为构造器循环依赖与setter循环依赖两种情况。构造器循环依赖不可解决,Spring会在检测到此类依赖时抛出`BeanCurrentlyInCreationException`异常。setter循环依赖则通过缓存机制解决:利用三级缓存系统,其中一级缓存`singletonObjects`存放已完成的单例Bean;二级缓存`earlySingletonObjects`存放实例化但未完成属性注入的Bean;三级缓存`singletonFactories`存放创建这些半成品Bean的工厂。
|
29天前
|
缓存 Java 开发者
Spring循环依赖问题之Spring循环依赖如何解决
Spring循环依赖问题之Spring循环依赖如何解决
|
29天前
|
缓存 Java Spring
Spring循环依赖问题之Spring不支持构造器内的强依赖注入如何解决
Spring循环依赖问题之Spring不支持构造器内的强依赖注入如何解决
|
29天前
|
Java Spring
Spring循环依赖问题之构造器内的循环依赖如何解决
Spring循环依赖问题之构造器内的循环依赖如何解决
|
29天前
|
Java Spring 容器
Spring循环依赖问题之两个不同的Bean A,导致抛出异常如何解决
Spring循环依赖问题之两个不同的Bean A,导致抛出异常如何解决
|
29天前
|
存储 缓存 Java
Spring循环依赖问题之循环依赖异常如何解决
Spring循环依赖问题之循环依赖异常如何解决
|
29天前
|
XML Java 数据格式
循环依赖问题之创建Bean的过程中发生异常,Spring会如何处理
循环依赖问题之创建Bean的过程中发生异常,Spring会如何处理
|
3月前
|
缓存 Java 开发工具
【spring】如何解决循环依赖
【spring】如何解决循环依赖
179 56