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

时间:2021-02-14 00:50:28


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.

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

 

2:分析原因

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

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

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

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

 

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  class="org.xyz.ServiceDependent1" lazy-init="true">
<constructor-arg ref="Service"/>
</bean>

<bean 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;