Spring解决循环依赖

一、什么是循环依赖

循环依赖就是循环引用,就是两个或多个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);
    }
});

具体步骤如下:

  1. Spring容器创建单例“A”bean,首先根据无参构造器创建bean,并暴露一个“ObjectFactory”用于返回一个提前暴露一个创建中的bean,并将“A”标识符放到“当前创建bean池”,然后进行setter注入“B“。
  2. Spring容器创建单例“B”bean,首先根据无参构造器创建bean,并暴露一个“ObjectFactory”用于返回一个提前暴露一个创建中的bean,并将“B”标识符放到“当前创建bean池”,然后进行setter注入“C“。
  3. Spring容器创建单例“C”bean,首先根据无参构造器创建bean,并暴露一个“ObjectFactory”用于返回一个提前暴露一个创建中的bean,并将“C”标识符放到“当前创建bean池”,然后进行setter注入“A“。进行注入”A“时由于提前暴露了”ObjectFactory“工厂,从而使用它返回提前暴露一个创建中的bean。
  4. 最后在依赖注入”B“和”A“,完成setter注入。
原文地址:https://www.cnblogs.com/jinchengll/p/11847284.html