一、什么是循环依赖
循环依赖就是循环引用,就是两个或多个bean相互之间的持有对方,比如A引用B、B引用C、C引用A,则它们最终反映为一个环。此处不是循环调用,循环调用是方法之间环调用。循环依赖如下图:
二、Spring如何解决循环依赖
Spring容器循环依赖包括构造器循环依赖和setter循环依赖,那Spring容器如何解决循环依赖呢?首先定义循环引用类:
public class A
{
private B b;
public B getB()
{
return b;
}
public void setB(B b)
{
this.b = b;
}
}
public class B
{
private C c;
public C getC()
{
return c;
}
public void setC(C c)
{
this.c = c;
}
}
public class C
{
private A a;
public A getA()
{
return a;
}
public void setA(A a)
{
this.a = a;
}
}
在Spring中将循环依赖的处理分成了3种情况
1、构造器循环依赖
表示通过构造器注入构成的循环依赖,此以来是无法解决的,只能抛出BeanCurrentlyInCreationException异常表示循环依赖。
如在创建A类时,构造器需要B类,那将去创建B,在创建B类时又发现需要C类,则又去创建C类,最终在C类中发现有需要A,从而形成一个环,没办法创建。
Spring容器将每一个正在创建的bean标识符放在一个“当前创建bean池”中,bean标识符在创建过程中将一直保持在这个池中,因此如果在创建bean过程中发现自己已经在“当前创建bean池”中,将抛出BeanCurrentlyInCreationException异常表示循环依赖;而对于创建完毕的bean将从“当前创建bean池”中清除掉。
2、setter循环依赖
表示通过settte注入方式构成的循环依赖。对于setter注入造成的依赖是通过Spring容器提前暴露刚完成构造器注入但未完成其他步骤(如setter注入)的bean来完成的,而且只能解决单例作用域的bean循环依赖。通过提前暴露一个单例工厂方法,从而使其他bean能引用到该bean,如下代码所示:
addSingletonFactory(beanName, new ObjectFactory() {
public Object getObject() throws BeansException {
return getEarlyBeanReference(beanName, mbd, bean);
}
});
具体步骤如下:
- Spring容器创建单例“A”bean,首先根据无参构造器创建bean,并暴露一个“ObjectFactory”用于返回一个提前暴露一个创建中的bean,并将“A”标识符放到“当前创建bean池”,然后进行setter注入“B“。
- Spring容器创建单例“B”bean,首先根据无参构造器创建bean,并暴露一个“ObjectFactory”用于返回一个提前暴露一个创建中的bean,并将“B”标识符放到“当前创建bean池”,然后进行setter注入“C“。
- Spring容器创建单例“C”bean,首先根据无参构造器创建bean,并暴露一个“ObjectFactory”用于返回一个提前暴露一个创建中的bean,并将“C”标识符放到“当前创建bean池”,然后进行setter注入“A“。进行注入”A“时由于提前暴露了”ObjectFactory“工厂,从而使用它返回提前暴露一个创建中的bean。
- 最后在依赖注入”B“和”A“,完成setter注入。