Spring為何要用三級(jí)緩存來解決循環(huán)依賴問題
我們都知道Spring為了解決循環(huán)依賴使用了三級(jí)緩存
Spring三級(jí)緩存
一級(jí)緩存singletonObjects
用于保存BeanName和創(chuàng)建bean實(shí)例之間的關(guān)系,beanName -> bean instance
private final Map<String, Object> singletonObjects = new ConcurrentHashMap(256);
二級(jí)緩存earlySingletonObjects
保存提前曝光的單例bean對(duì)象
private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);
三級(jí)緩存singletonFactories
保存beanName和創(chuàng)建bean實(shí)例之間的關(guān)系,與singletonObjects不同的地方在于,當(dāng)一個(gè)單例bean被放到這里面后,bean在創(chuàng)建過程中,可以通過getBean方法獲取到,目的是用來檢測(cè)循環(huán)引用
private final Map<String, Object> singletonFactories = new HashMap(16);
在創(chuàng)建bean的時(shí)候,首先從緩存中獲取單例的bean,這個(gè)緩存就是singletonObjects,如果獲取不到且bean正在創(chuàng)建中,就再從earlySingletonObjects中獲取,如果還是獲取不到且允許從singletonFactories中通過getObject拿到對(duì)象,就從singletonFactories中獲取,如果獲取到了就存入earlySingletonObjects并從singletonFactories中移除。
為什么要使用三級(jí)緩存?
一級(jí)緩存
首先我們看看一級(jí)緩存行不行,如果只留第一級(jí)緩存,那么單例的Bean都存在singletonObjects 中,Spring循環(huán)依賴主要基于Java引用傳遞,當(dāng)獲取到對(duì)象時(shí),對(duì)象的field或者屬性可以延后設(shè)置,理論上可以,但是如果延后設(shè)置出了問題,就會(huì)導(dǎo)致完整的Bean和不完整的Bean都在一級(jí)緩存中,這個(gè)引用時(shí)就有空指針的可能,所以一級(jí)緩存不行,至少要有singletonObjects 和earlySingletonObjects 兩級(jí)。
兩級(jí)緩存
那么我們?cè)倏纯磧杉?jí)緩存行不行
現(xiàn)在有A的field或者setter依賴B的實(shí)例對(duì)象,同時(shí)B的field或者setter依賴了A的實(shí)例,A首先開始創(chuàng)建,并將自己暴露到 earlySingletonObjects 中,開始填充屬性,此時(shí)發(fā)現(xiàn)自己依賴B的屬性,嘗試去get(B),發(fā)現(xiàn)B還沒有被創(chuàng)建,所以開始創(chuàng)建B,在進(jìn)行屬性填充時(shí)初始化A,就從earlySingletonObjects 中獲取到了實(shí)例化但沒有任何屬性的A,B拿到A后完成了初始化階段,將自己放到singletonObjects中,此時(shí)返回A,A拿到B的對(duì)象繼續(xù)完成初始化,完成后將自己放到singletonObjects中,由A與B中所表示的A的屬性地址是一樣的,所以A的屬性填充完后,B也獲取了A的屬性,這樣就解決了循環(huán)的問題。
似乎完美解決,如果就這么使用的話也沒什么問題,但是再加上AOP情況就不同了,被AOP增強(qiáng)的Bean會(huì)在初始化后代理成為一個(gè)新的對(duì)象,也就是說:
如果有AOP,A依賴于B,B依賴于A,A實(shí)例化完成暴露出去,開始注入屬性,發(fā)現(xiàn)引用B,B開始實(shí)例化,使用A暴露的對(duì)象,初始化完成后封裝成代理對(duì)象,A再將代理后的B注入,再做代理,那么代理A中的B就是代理后的B,但是代理后的B中的A是沒用代理的A。
顯然這是不對(duì)的,所以在Spring中存在第三級(jí)緩存,在創(chuàng)建對(duì)象時(shí)判斷是否是單例,允許循環(huán)依賴,正在創(chuàng)建中,就將其從earlySingletonObjects中移除掉,并在singletonFactories放入新的對(duì)象,這樣后續(xù)再查詢beanName時(shí)會(huì)走到singletonFactory.getObject(),其中就會(huì)去調(diào)用各個(gè)beanPostProcessor的getEarlyBeanReference方法,返回的對(duì)象就是代理后的對(duì)象。
public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory implements AutowireCapableBeanFactory { // Eagerly cache singletons to be able to resolve circular references // even when triggered by lifecycle interfaces like BeanFactoryAware. boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences && isSingletonCurrentlyInCreation(beanName)); if (earlySingletonExposure) { if (logger.isDebugEnabled()) { logger.debug("Eagerly caching bean '" + beanName + "' to allow for resolving potential circular references"); } addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean)); } /** * Add the given singleton factory for building the specified singleton * if necessary. * <p>To be called for eager registration of singletons, e.g. to be able to * resolve circular references. * @param beanName the name of the bean * @param singletonFactory the factory for the singleton object */ protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) { Assert.notNull(singletonFactory, "Singleton factory must not be null"); synchronized (this.singletonObjects) { if (!this.singletonObjects.containsKey(beanName)) { this.singletonFactories.put(beanName, singletonFactory); this.earlySingletonObjects.remove(beanName); this.registeredSingletons.add(beanName); } } }
總結(jié)
到此這篇關(guān)于Spring為何要用三級(jí)緩存來解決循環(huán)依賴問題的文章就介紹到這了,更多相關(guān)Spring用三級(jí)緩存解決循環(huán)依賴內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringCloud Gateway網(wǎng)關(guān)功能介紹與使用
SpringCloud Gateway 是 Spring Cloud 的一個(gè)全新項(xiàng)目,它旨在為微服務(wù)架構(gòu)提供一種簡(jiǎn)單有效的統(tǒng)一的 API 路由管理方式。這篇文章主要介紹了SpringCloud Gateway網(wǎng)關(guān)作用,需要的朋友可以參考下2022-12-12基于Java實(shí)現(xiàn)互聯(lián)網(wǎng)實(shí)時(shí)聊天系統(tǒng)(附源碼)
Netty?是一個(gè)利用?Java?的高級(jí)網(wǎng)絡(luò)的能力,隱藏其背后的復(fù)雜性而提供一個(gè)易于使用的?API?的客戶端/服務(wù)器框架。本文將利用它實(shí)現(xiàn)互聯(lián)網(wǎng)實(shí)時(shí)聊天系統(tǒng),感興趣的可以了解一下2022-09-09Java操作數(shù)據(jù)庫(行級(jí)鎖,for update)
這篇文章主要介紹了Java操作數(shù)據(jù)庫(行級(jí)鎖,for update),文章圍繞Java操作數(shù)據(jù)庫的相關(guān)資料展開詳細(xì)內(nèi)容,需要的小伙伴可以參考一下,希望對(duì)你有所幫助2021-12-12mybatis中如何實(shí)現(xiàn)一個(gè)標(biāo)簽執(zhí)行多個(gè)sql語句
這篇文章主要介紹了mybatis中如何實(shí)現(xiàn)一個(gè)標(biāo)簽執(zhí)行多個(gè)sql語句問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-04-04Fluent Mybatis實(shí)際開發(fā)中的優(yōu)勢(shì)對(duì)比
本文給大家介紹如何通過IQuery和IUpdate定義強(qiáng)大的動(dòng)態(tài)SQL語句,給大家分享Fluent Mybatis實(shí)際開發(fā)中的優(yōu)勢(shì)講解,感興趣的朋友一起看看吧2021-08-08如何在Java SpringBoot項(xiàng)目中配置動(dòng)態(tài)數(shù)據(jù)源你知道嗎
這篇文章主要介紹了SpringBoot如何在運(yùn)行時(shí)動(dòng)態(tài)添加數(shù)據(jù)源,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2021-09-09Maven中的dependencyManagement 實(shí)例詳解
dependencyManagement的中文意思就是依賴關(guān)系管理,它就是為了能通更好統(tǒng)一管理項(xiàng)目的版本號(hào)和各種jar版本號(hào),可以更加方便升級(jí),解決包沖突問題,這篇文章主要介紹了Maven中的dependencyManagement 實(shí)例詳解,需要的朋友可以參考下2024-02-02Spring對(duì)靜態(tài)變量無法注入的解決方案
這篇文章主要介紹了使用Spring對(duì)靜態(tài)變量無法注入的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。2021-07-07