spring循环依赖通俗总结如果循环依赖的几个类都是构造注入,会出现问题。如果循环以来的类通过setter注入,只
spring循环依赖通俗总结
如果循环依赖的几个类都是构造注入,会出现问题。如果循环以来的类通过setter注入,只要有一个类的scope的单例的,就没有问题。如果采用?depends-on而造成循环以来,会出现问题!如果循环依赖的类都是单例的,A、B两个类循环setter依赖,A类先被初始化。两个类都有init方法,并且互相引用对方。
a. B类init中 B调用A.say(),A.say()中调用B.init(),会抛空指针错误!因为A类中B类还没有被注入。
b. A类init中 A调用B.say(),B.say()中调用A.init(),会栈溢出。
? ? ? ? ? ? ? ? c. 否则无问题
?
?
最后引一段ITPUB上找到的原理。
?
?
</bean>
<bean id="circleB" ref="circleC"/>
</bean>
<bean id="circleC" ref="circleA"/>
</bean>
2)写段测试代码(cn.javass.spring.chapter3.CircleTest)测试一下吧:
java代码:查看复制到剪贴板打印
@Test(expected = BeanCurrentlyInCreationException.class)
public void testCircleByConstructor() throws Throwable {
try {
new ClassPathXmlApplicationContext("chapter3/circleInjectByConstructor.xml");
}
catch (Exception e) {
//因为要在创建circle3时抛出;
Throwable e1 = e.getCause().getCause().getCause();
throw e1;
}
}
让我们分析一下吧:
1、Spring容器创建“circleA” Bean,首先去“当前创建Bean池”查找是否当前Bean正在创建,如果没发现,则继续准备其需要的构造器参数“circleB”,并将“circleA” 标识符放到“当前创建Bean池”;
2、Spring容器创建“circleB” Bean,首先去“当前创建Bean池”查找是否当前Bean正在创建,如果没发现,则继续准备其需要的构造器参数“circleC”,并将“circleB” 标识符放到“当前创建Bean池”;
3、Spring容器创建“circleC” Bean,首先去“当前创建Bean池”查找是否当前Bean正在创建,如果没发现,则继续准备其需要的构造器参数“circleA”,并将“circleC” 标识符放到“当前创建Bean池”;
4、到此为止Spring容器要去创建“circleA”Bean,发现该Bean 标识符在“当前创建Bean池”中,因为表示循环依赖,抛出BeanCurrentlyInCreationException。
二、setter循环依赖:表示通过setter注入方式构成的循环依赖。
对于setter注入造成的依赖是通过Spring容器提前暴露刚完成构造器注入但未完成其他步骤(如setter注入)的Bean来完成的,而且只能解决单例作用域的Bean循环依赖。
如下代码所示,通过提前暴露一个单例工厂方法,从而使其他Bean能引用到该Bean。
java代码:查看复制到剪贴板打印
addSingletonFactory(beanName, new ObjectFactory() {
public Object getObject() throws BeansException {
return getEarlyBeanReference(beanName, mbd, bean);
}
});
具体步骤如下:
1、Spring容器创建单例“circleA” Bean,首先根据无参构造器创建Bean,并暴露一个“ObjectFactory ”用于返回一个提前暴露一个创建中的Bean,并将“circleA” 标识符放到“当前创建Bean池”;然后进行setter注入“circleB”;
2、Spring容器创建单例“circleB” Bean,首先根据无参构造器创建Bean,并暴露一个“ObjectFactory”用于返回一个提前暴露一个创建中的Bean,并将“circleB” 标识符放到“当前创建Bean池”,然后进行setter注入“circleC”;
3、Spring容器创建单例“circleC” Bean,首先根据无参构造器创建Bean,并暴露一个“ObjectFactory ”用于返回一个提前暴露一个创建中的Bean,并将“circleC” 标识符放到“当前创建Bean池”,然后进行setter注入“circleA”;进行注入“circleA”时由于提前暴露了“ObjectFactory”工厂从而使用它返回提前暴露一个创建中的Bean;
4、最后在依赖注入“circleB”和“circleA”,完成setter注入。
对于“prototype”作用域Bean,Spring容器无法完成依赖注入,因为“prototype”作用域的Bean,Spring容器不进行缓存,因此无法提前暴露一个创建中的Bean。
java代码:查看复制到剪贴板打印
<!-- 定义Bean配置文件,注意scope都是“prototype”-->
<bean id="circleA" scope="prototype">
<property name="circleB" ref="circleB"/>
</bean>
<bean id="circleB" scope="prototype">
<property name="circleC" ref="circleC"/>
</bean>
<bean id="circleC" scope="prototype">
<property name="circleA" ref="circleA"/>
</bean>
java代码:查看复制到剪贴板打印
//测试代码cn.javass.spring.chapter3.CircleTest
@Test(expected = BeanCurrentlyInCreationException.class)
public void testCircleBySetterAndPrototype () throws Throwable {
try {
ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext(
"chapter3/circleInjectBySetterAndPrototype.xml");
System.out.println(ctx.getBean("circleA"));
}
catch (Exception e) {
Throwable e1 = e.getCause().getCause().getCause();
throw e1;
}
}
对于“singleton”作用域Bean,可以通过“setAllowCircularReferences(false);”来禁用循环引用:
java代码:查看复制到剪贴板打印
@Test(expected = BeanCurrentlyInCreationException.class)
public void testCircleBySetterAndSingleton2() throws Throwable {
try {
ClassPathXmlApplicationContext ctx =
new ClassPathXmlApplicationContext();
ctx.setConfigLocation("chapter3/circleInjectBySetterAndSingleton.xml");
ctx.refresh();
}
catch (Exception e) {
Throwable e1 = e.getCause().getCause().getCause();
throw e1;
}
}