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;