咱們知道 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而後返回。