循环依赖是指多个类循环嵌套引用,如:A类引用B类,B类引用C类,C类引用A类。
第一种:构造器参数循环依赖
Spring容器会将每一个正在创建的Bean标识放在一个“当前创建Bean池中,Bean标识符在创建过程中将一直保持在这个池中。
如果在创建Bean过程中发现自己已经在“当前创建Bean池”里时将抛出BeanCurrentlyInCreationException异常表示循环依赖;而创建完成的Bean将从“当前创建Bean池”中清除掉。
看如下实例:
![](https://img.haomeiwen.com/i14944898/d2ec7a1a605d1344.png)
![](https://img.haomeiwen.com/i14944898/e2ba5f8740677e28.png)
![](https://img.haomeiwen.com/i14944898/a55aa5fccce8282a.png)
看配置文件的构造器参数配置:
<bean id="a" class="com.soecode.lyf.test.StudentA" >
<constructor-arg index="0" ref="b"/>
</bean>
<bean id="b" class="com.soecode.lyf.test.StudentB" >
<constructor-arg index="0" ref="c" >
</bean>
<bean id="c" class="com.soecode.lyf.test.StudentC" >
<constructor-arg index="0" ref="a" >
</bean>
这三个就是StudentA类中有StudentB类,StudentB类中有StudentC类,StudentC类中有StudentA类。
![](https://img.haomeiwen.com/i14944898/20d2dc83003ef04c.png)
第二种:setter方式单列,默认方式
![](https://img.haomeiwen.com/i14944898/fa0d53de1187cf43.png)
看如下实例:
![](https://img.haomeiwen.com/i14944898/2d7256e28172ebe4.png)
![](https://img.haomeiwen.com/i14944898/49bde7d3f91bb6b3.png)
![](https://img.haomeiwen.com/i14944898/64e63938558ecc52.png)
setter 方式注入 scope默认为singleton
<bean id="a" class="com.soecode.lyf.test.StudentA" >
<property name="studentB" ref="b">
</bean>
<bean id="b" class="com.soecode.lyf.test.StudentB" >
<property name="studentC" ref="c" >
</bean>
<bean id="c" class="com.soecode.lyf.test.StudentC" >
<property name="studentA" ref="a" >
</bean>
执行结果:
![](https://img.haomeiwen.com/i14944898/13f68a2d5bbaeee9.png)
为什么用setter方式就不报错了呢?
/** Cache of singleton factories: bean name --> ObjectFactory(单例的工厂Bean缓存集合) */、privatefinalMap singletonFactories =newHashMap(16);
1.创建StudentA a单列时,首先无参构造创建,并暴露到singletonFactories中,并将a 标志符放到当前正在创建Bean池, 然后进行setter出入StudentB b。
2.创建StudentB b单列时,首先无参构造创建,并暴露到singletonFactories中,并将b 标志符放到当前正在创建Bean池, 然后进行setter出入StudentC c。
3.创建StudentC c单列时,首先无参构造创建,并暴露到singletonFactories中,并将b 标志符放到当前正在创建Bean池, 然后进行setter出入StudentA a。在注入 a 时,由于提前暴露在singletonFactories集合中了,利用它就可以取到 a 正在创建的Bean对象。
4. 最后依赖注入StudentB、StudentA
第三种:setter方式 原型注入
看配置文件:
![](https://img.haomeiwen.com/i14944898/d14202243f3bee22.png)
scope=”property“ 意思是每一次请求都会创建一个实例对象。
两者的区别是:有状态的bean都使用property作用域,无状态的一般都使用singleton单列作用域。
看运行之后的控制台输出:
![](https://img.haomeiwen.com/i14944898/5e8f7f370e9586b9.png)
总结:为什么setter方式 singleton单列会成功,而setter方式 property原型会报BeanCurrentlyInCreationException?
因为单列的时候,会将bean放在缓存中,可以提前暴露此接口。
而property原型不会放入缓存中,无法提前暴露。
网友评论