Spring 三种方式的循环依赖

2019-04-02  本文已影响0人  loveinthesweet

循环依赖是指多个类循环嵌套引用,如: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原型不会放入缓存中,无法提前暴露。

上一篇下一篇

猜你喜欢

热点阅读