您好,登錄后才能下訂單哦!
這篇文章主要講解了“怎么理解Spring源碼循環(huán)依賴”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“怎么理解Spring源碼循環(huán)依賴”吧!
Spring是怎么去解決循環(huán)依賴的
1.什么是循環(huán)依賴
這個詞,阿粉聽到的時候,肯定和大家的反應一樣的,循環(huán),依賴,那是不是 A 引用了 B ,而此時 B 引用了 C,而 C 呢又引用了A,于是一個三角戀的關系出現(xiàn)了。
那么用代碼來表示的話,是怎么表示的呢?
public class ClassTestA { private ClassTestB classTestB; public void a(){ classTestB.b(); } public ClassTestB getClassTestB() { return classTestB; } private void setClassTestB(ClassTestB classTestB){ this.classTestB = classTestB; } } public class ClassTestB { private ClassTestC classTestC; public void b(){ classTestC.c(); } public ClassTestC getClassTestC() { return classTestC; } private void setClassTestC(ClassTestC classTestC){ this.classTestC = classTestC; } } public class ClassTestC { private ClassTestA classTestA; public void c(){ classTestA.a(); } public ClassTestA getClassTestA() { return classTestA; } private void setClassTestA(ClassTestA classTestA){ this.classTestA = classTestA; } }
2.循環(huán)依賴會出現(xiàn)什么問題
在阿粉的印象中,循環(huán)依賴最直接的問題就是會出現(xiàn)在對象的實例化上面,創(chuàng)建對象的時候,如果在Spring的配置中加入這種 A 依賴 B ,B 依賴 C,C 依賴 A 的話,那么最終創(chuàng)建 A 的實例對象的時候,會出現(xiàn)錯誤。
而如果這種循環(huán)調用的依賴不去終結掉他的話,那么就相當于一個死循環(huán),就像阿粉前幾天的在維護那個 “十六年”之前的項目的時候,各種內存溢出,表示內心很壓抑呀。
而 Spring 中也將循環(huán)依賴的處理分成了不同的幾種情況,阿粉帶大家來看一下吧。
3.Spring循環(huán)依賴處理一 (構造器循環(huán)依賴)
構造器循環(huán)依賴的意思就是說,通過構造及注入構成的循環(huán)依賴,而這種依賴的話,是沒有辦法解決的,如果你敢強行依賴,不要意思,出現(xiàn)了你久違的異常 BeanCurrentlyInCreationException 出現(xiàn)這個異常的時候,就是表示循環(huán)依賴的問題。
相信大家出現(xiàn)異常的時候,在看不懂為什么的時候,第一時間,復制異常信息,放在百度,或者Google上面查詢一下,BeanCurrentlyInCreationException 放在百度上,一目了然。
而在 Spring 的配置文件中,如果這么配置 A ,B ,C 的循環(huán)依賴的時候,在創(chuàng)建 A 的時候,發(fā)現(xiàn),構造器需要 B 類,然后去創(chuàng)建 B ,而創(chuàng)建 B 的時候,發(fā)現(xiàn)又需要 C ,然后去創(chuàng)建 C ,創(chuàng)建的時候發(fā)現(xiàn),竟然需要 A ,于是又掉頭回去了,于是就形成了一個閉環(huán),沒有辦法創(chuàng)建。
<beans> <bean id="ClassTestA" class="com.yldlsy.ClassTestA" > <constructor-arg index="0" ref="ClassTestB" /> </bean> <bean id="ClassTestB" class="com.yldlsy.ClassTestB" > <constructor-arg index="0" ref="ClassTestC" /> </bean> <bean id="ClassTestC" class="com.yldlsy.ClassTestC" > <constructor-arg index="0" ref="ClassTestA" /> </bean> </beans>
而在這種情況下,Spring實例化bean是通過ApplicationContext.getBean()方法來進行的。如果要獲取的對象依賴了另一個對象,那么其首先會創(chuàng)建當前對象,然后通過遞歸的調用ApplicationContext.getBean()方法來獲取所依賴的對象,最后將獲取到的對象注入到當前對象中。而和剛才阿粉說的一樣,創(chuàng)建了閉環(huán),所以就沒有辦法創(chuàng)建了。
4.Spring循環(huán)依賴處理二(setter循環(huán)依賴)
setter循環(huán)注入是指通過setter注入方式構成的循環(huán)依賴。而這種方式,是Spring可以進行解決的。
而對于這種使用setter注入造成的依賴是通過Spring容器來提前暴露剛完成的構造注入器的bean來完成的,但是這時候還沒有完成其他的步驟的時候。
這個時候我們就需要提前暴露出來一個單例的工廠方法,讓其他的bean來引用這個bean,
addSingletonFactory(beanName,new ObjectFactory(){ public Object getObject() throws BeanException{ return getEarlyBeanReference(beanName,mbd,bean) } })
Spring在創(chuàng)建 A 的時候,根據(jù)無參構造來創(chuàng)建 A,并且暴露出 ObjectFactory 用來返回一個提前暴露好的 bean 然后再進行setter來注入,
同理的B和C都是這個樣子的,這個時候就能完成setter注入了。
5.Spring循環(huán)依賴處理三(作用域循環(huán)依賴)
阿粉帶大家看一下這個配置方式
<bean id="ClassTestA" class="com.yldlsy.ClassTestA" scope="singleton" > <property name="ClassTestB" ref="ClassTestB" /> </bean> <bean id="ClassTestA" class="com.yldlsy.ClassTestA" scope="singleton" > <property name="ClassTestB" ref="ClassTestB" /> </bean> <bean id="ClassTestA" class="com.yldlsy.ClassTestA" scope="singleton" > <property name="ClassTestB" ref="ClassTestB" /> </bean>
而對于 “singleton”作用域的話,他是可以通過“setAllowCircularReference(false)”這種方式來進制循環(huán)依賴的。
而且也是有缺陷的,這種方式只能解決單例作用域的bean循環(huán)依賴。
而Spring解決循環(huán)依賴的話,大家肯定會好奇,說是三級緩存,那么請找到你的Spring的源碼
org.springframework.beans.factory.support.AbstractBeanFactory#doGetBean
感謝各位的閱讀,以上就是“怎么理解Spring源碼循環(huán)依賴”的內容了,經過本文的學習后,相信大家對怎么理解Spring源碼循環(huán)依賴這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經查實,將立刻刪除涉嫌侵權內容。