循环依赖是指多个类循环嵌套引用,如:A类引用B类,B类引用C类,C类引用A类。
第一种:构造器参数循环依赖
Spring容器会将每一个正在创建的Bean标识放在一个“当前创建Bean池中,Bean标识符在创建过程中将一直保持在这个池中。
如果在创建Bean过程中发现自己已经在“当前创建Bean池”里时将抛出BeanCurrentlyInCreationException异常表示循环依赖;而创建完成的Bean将从“当前创建Bean池”中清除掉。
看如下实例:
看配置文件的构造器参数配置:
<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类。
第二种:setter方式单列,默认方式
看如下实例:
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>
执行结果:
为什么用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方式 原型注入
看配置文件:
scope=”property“ 意思是每一次请求都会创建一个实例对象。
两者的区别是:有状态的bean都使用property作用域,无状态的一般都使用singleton单列作用域。
看运行之后的控制台输出:
总结:为什么setter方式 singleton单列会成功,而setter方式 property原型会报BeanCurrentlyInCreationException?
因为单列的时候,会将bean放在缓存中,可以提前暴露此接口。
而property原型不会放入缓存中,无法提前暴露。