spring循環(huán)注入異常問題的解決方案
今天在做項目的時候突然遇到一個問題:啟動服務(wù)器的時候spring
沒報錯,可是當我訪問某個頁面的時候spring報Request bean is currently in creation: is there an unresolvable circular reference
的錯誤,后來查了些資源終于找出來了原因,這里和大家分享一下;
首先產(chǎn)生這個錯誤的原因是因為spring循環(huán)注入了,什么是循環(huán)注入?舉個列子我有一個類A,A有一個構(gòu)造器里面的參數(shù)是類B,然后類B里面有個構(gòu)造器參數(shù)是類C,類C里面有個構(gòu)造器參數(shù)是類A,就是我們會發(fā)現(xiàn)其實引用循環(huán)了A 里面有B的引用,B里面有C的引用,C里面又有A的引用。
循環(huán)依賴又分為構(gòu)造器循環(huán)依賴和set
循環(huán)依賴:
首先講一下構(gòu)造器的循環(huán)依賴:
public class A { public A(B b) { } }
public class B { public B(C c) { } }
public class C { public C(A a) { } }
當我們用spring來加載A的時候spring的流程是這樣的:
1:spring創(chuàng)建A首先去當前創(chuàng)建池中去查找當前A是否在創(chuàng)建,如果發(fā)明沒有創(chuàng)建則準備其構(gòu)造器需要的參數(shù)B,然后把創(chuàng)建A的標識放入當前創(chuàng)建池中。
2:spring創(chuàng)建B首先去當前創(chuàng)建池中去查找當前B是否在創(chuàng)建,如果發(fā)現(xiàn)沒有創(chuàng)建則準備其構(gòu)造器需要的參數(shù)C,然后把創(chuàng)建B的標識放入當前創(chuàng)建池中。
3:spring創(chuàng)建C首先去當前創(chuàng)建池中去查找當前C是否在創(chuàng)建,如果發(fā)現(xiàn)沒有創(chuàng)建則準備其構(gòu)造器需要的參數(shù)A,然后把創(chuàng)建C的標識放入當前創(chuàng)建池中。
4:spring創(chuàng)建C需要的A,這個時候會發(fā)現(xiàn)在當前創(chuàng)建池中已經(jīng)有A的標識,A正在創(chuàng)建中則拋出BeanCurrentlyInCreationException。
構(gòu)造器的循環(huán)注入是沒有辦法解決的,所以只能我們避免.
接下來看下set方式的循環(huán)注入:
set方式的循環(huán)注入分2種情況,第一種情況是可以解決的循環(huán)注入就是單列情況下。第二種情況就是無法解決的循環(huán)注入就是多列情況下,下面分析一下原因:
先看第一種情況,還是拿上面的ABC3個類來說明問題,只不過這次不是構(gòu)造器里面的參數(shù),而是換成他們的成員變量,然后通過set方式類注入,這里代碼就不寫了直接講下:
單列下set方式的注入流程是這樣的:
1:spring創(chuàng)建A,首先根據(jù)其無參構(gòu)造器創(chuàng)建一個對象A,然后提前暴露出創(chuàng)建出來的這個A對象,然后再當前的創(chuàng)建池中放入創(chuàng)建A的標識,然后進行set方法注入B。
2:spring創(chuàng)建B,首先根據(jù)其無參構(gòu)造器創(chuàng)建一個對象B,然后提前暴露出創(chuàng)建出來的這個B對象,然后在當前的創(chuàng)建池中放入創(chuàng)建B的標識,然后進行set方法的注入C。
3:spring創(chuàng)建C,首先根據(jù)其無參構(gòu)造器創(chuàng)建一個對象C,然后提前暴露出創(chuàng)建處理的這個C對象,然后在當前的創(chuàng)建池中放入創(chuàng)建C的標識,然后進行set方法的注入A。
4:在第三步注入A的時候由于提前暴露出來了創(chuàng)建出來的A對象所以不會報BeanCurrentlyInCreationException的錯誤。
多列下set方式的循環(huán)注入不能解決的原因是在多列的情況下,當創(chuàng)建對象的時候spring不會提前暴露創(chuàng)建處理的對象A,這樣的話則會和構(gòu)造器循環(huán)注入出現(xiàn)一樣的情況最終導致報錯。
解決辦法:
去掉最后一層的注入或者中間某一層的注入。通過spring getBean
的方式去獲取對象
講完了。講的不對的地方謝謝提出來。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請查看下面相關(guān)鏈接
- Spring循環(huán)依賴正確性及Bean注入的順序關(guān)系詳解
- 詳解Spring-bean的循環(huán)依賴以及解決方式
- 詳解Spring循環(huán)依賴的解決方案
- 深入理解Spring中的循環(huán)依賴
- Spring循環(huán)依賴的三種方式(推薦)
- 詳解Spring Bean的循環(huán)依賴解決方案
- spring boot啟動時mybatis報循環(huán)依賴的錯誤(推薦)
- 淺談Spring解決循環(huán)依賴的三種方式
- 詳解在springmvc中解決FastJson循環(huán)引用的問題
- Spring中DAO被循環(huán)調(diào)用的時候數(shù)據(jù)不實時更新的解決方法
相關(guān)文章
springboot使用redis緩存亂碼(key或者value亂碼)的解決
在通過springboot緩存數(shù)據(jù)的時候,發(fā)現(xiàn)key是一堆很不友好的東西,本文主要介紹了springboot使用redis緩存亂碼(key或者value亂碼)的解決,感興趣的可以了解一下2023-11-11Hibernate中使用HQLQuery查詢?nèi)繑?shù)據(jù)和部分數(shù)據(jù)的方法實例
今天小編就為大家分享一篇關(guān)于Hibernate中使用HQLQuery查詢?nèi)繑?shù)據(jù)和部分數(shù)據(jù)的方法實例,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-03-03Java利用MessageFormat實現(xiàn)短信模板的匹配
這篇文章主要介紹了Java利用MessageFormat實現(xiàn)短信模板的匹配,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-06-06Java實戰(zhàn)之小米交易商城系統(tǒng)的實現(xiàn)
這篇文章將利用Java實現(xiàn)小米交易商城系統(tǒng),文中采用的技術(shù)有:JSP?、Spring、SpringMVC、MyBatis等,感興趣的小伙伴可以跟隨小編一起學習一下2022-04-04JavaWeb實戰(zhàn)之編寫單元測試類測試數(shù)據(jù)庫操作
這篇文章主要介紹了JavaWeb實戰(zhàn)之編寫單元測試類測試數(shù)據(jù)庫操作,文中有非常詳細的代碼示例,對正在學習javaweb的小伙伴們有很大的幫助,需要的朋友可以參考下2021-04-04迅速學會@ConfigurationProperties的使用操作
這篇文章主要介紹了迅速學會@ConfigurationProperties的使用,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-10-10Java實戰(zhàn)之校園外賣點餐系統(tǒng)的實現(xiàn)
這篇文章主要介紹了如何利用Java實現(xiàn)簡易的校園外賣點餐系統(tǒng),文中采用的技術(shù)有:JSP、Spring、SpringMVC、MyBatis 等,感興趣的可以了解一下2022-03-03