Spring bean循環依賴問題,與解決方案。

前言

        咱們知道 Spring 能夠是懶加載的,就是當真正使用到 Bean 的時候才實例化 Bean。固然也不全是這樣,例如配置 Bean 的 lazy-init 屬性,能夠控制 Spring 的加載時機。如今機器的性能、內存等都比較高,基本上也不使用懶加載,在容器啓動時候來加載bean,啓動時間稍微長一點兒,這樣在實際獲取 bean 供業務使用時,就能夠減輕很多負擔,這個後面再作分析。 咱們使用到 Bean 的時候,最直接的方式就是從 Factroy 中獲取,這個就是加載 Bean 實例的源頭。java

        最近發現一個問題,一些大的公司(國內知名的boss級別公司就那麼幾家),在面試的過程當中,會問到一個基礎題:spring怎麼實現循環依賴,或者循環依賴的解決方案。今天主要就這個問題作下簡單的探討:web

        3個簡單的bean:TestA,TestB,TestC其中A包含B,B含C,C含A。而後把三類注入到spring容器中。操做以下:面試

A類:spring

public class TestA {
	
	private TestB testB;

	public TestB getTestB() {
		return testB;
	}

	public void setTestB(TestB testB) {
		this.testB = testB;
	}
}

B類:併發

public class TestB {
	
	private TestC testC;

	public TestC getTestC() {
		return testC;
	}

	public void setTestC(TestC testC) {
		this.testC = testC;
	}
}

C類:app

public class TestC {
	
	private TestA testA;

	public TestA getTestA() {
		return testA;
	}

	public void setTestA(TestA testA) {
		this.testA = testA;
	}
}

簡單的注入到spring中xml如:性能

<!-- 循環依賴測試 -->
<bean id="testA" class="com.xin.learn.xhyl.vo.TestA" scope="prototype">
	<property name="testB" ref="testB"></property>
</bean>
<bean id="testB" class="com.xin.learn.xhyl.vo.TestB" scope="prototype">
	<property name="testC" ref="testC"></property>
</bean>
<bean id="testC" class="com.xin.learn.xhyl.vo.TestC" scope="prototype">
	<property name="testA" ref="testA"></property>
</bean>

測試類:測試

public class TestMain {
	 public static void main(String[] args) {
		ApplicationContext context = new ClassPathXmlApplicationContext("classpath:spring/applicationContext.xml");
		System.out.println(context.getBean("testA", TestA.class));
	 }
}

若是你是web項目,運行項目不會報錯,可是當你引用的時候,或者運行測試類後發現報錯:ui

Error creating bean with name 'testA' defined in class path resource [spring/applicationContext.xml]: Cannot resolve reference to bean 'testB' while setting bean property 'testB'; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'testB' defined in class path resource [spring/applicationContext.xml]: Initialization of bean failed; nested exception is org.springframework.beans.ConversionNotSupportedException: Failed to convert property value of type 'com.xin.learn.xhyl.vo.TestB' to required type 'com.xin.learn.xhyl.vo.TestC' for property 'testC'; nested exception is java.lang.IllegalStateException: Cannot convert value of type [com.xin.learn.xhyl.vo.TestB] to required type [com.xin.learn.xhyl.vo.TestC] for property 'testC': no matching editors or conversion strategy found

大體意思是:在建立testA的時,設置屬性testB的時候不能引用testB。
由於這個時候的testB尚未被建立。this

解決:當把 scope的值改成singleton時,或者去掉scope(因spring默認的bean就是單例的),運行就正常了。緣由:
        Spring提供了EarlyBeanReference功能,首先Spring裏面有個名字爲singletonObjects的併發map用來存放全部實例化而且初始化好的bean,singletonFactories則用來存放須要解決循環依賴的bean信息(beanName,和一個回調工廠)。當實例化beanA時候會觸發getBean(「beanA」);首先看singletonObjects中是否有beanA有則返回,一開始確定沒有因此會實例化beanA,若是設置了allowCircularReferences=true(默認爲true)而且當前bean爲單件而且該bean目前在建立中,則初始化屬性前把該bean信息放入singletonFactories單件map裏面。而後對該實例進行屬性注入beanB,屬性注入時候會getBean(「beanB」) ,發現beanB 不在singletonObjects中,就會實例化beanB,而後放入singletonFactories,而後進行屬性注入beanA,而後觸發getBean(「beanA」);這時候會到(1)getSingleton返回實例化的beanA。到此beanB初始化完畢添加beanB 到singletonObjects而後返回,而後beanA 初始化完畢,添加beanA到singletonObjects而後返回。

相關文章
相關標籤/搜索