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