Spring為什么要用三級緩存解決循環(huán)依賴呢
1.什么是循環(huán)依賴
本文為了方便說明,先設(shè)置兩個業(yè)務(wù)層對象,命名為AService和BService。其中Spring是如何把一個Bean對象創(chuàng)建出來的,其生命周期如下:
構(gòu)造方法–> 不同對象 --> 注入依賴 -->初始化前 --> 初始化后–>放入單例池Map(一級緩存)—>Bean對象
Map的數(shù)據(jù)結(jié)構(gòu)為,key表示單例的名稱,而value是一個對象。其中單例池就是所謂的一級緩存。
循環(huán)依賴,如AService中依賴了一個BService,BService也依賴了AService,AService和BService相互依賴。這里換出現(xiàn)一個問題,但是Spring使用三級緩存給解決了。
首先我們要看的是為什么出現(xiàn)循環(huán)?
首先在AService中引入了@Componet注解,那AService的生命周期會交給Spring管理。AService生命周期如下,本文把該例子定義為示例1。
AService生命周期 1.實例化 --->AService的普通對象 2.填充BService(添加了AutoWired)-->從單例池中獲取BService(此時單例池中可能還沒存放BService的Bean)-->創(chuàng)建BService 當(dāng)單例池中可能還沒存放BService的Bean時,觸發(fā)創(chuàng)建BService生命周期,步驟和AService一致。如下所示: 2.1.實例化--->BService的普通對象 2.2.填充AService(添加了AutoWired)-->單例池(此時AAService也在創(chuàng)建中,還沒放入單例池,如果創(chuàng)建的話,就會出現(xiàn)循環(huán)依賴) 2.3.填充其他屬性 2.4.其他步驟(包括AOP) 2.5.加入到單例池中 3.填充其他屬性 4.其他步驟(包括AOP) 5.加入到單例池中
簡單解決示例1中的問題
主要是定義個Map存放第一步生成的普通對象。Map的名稱暫且為boziMap<beanName,普通對象>,示例2如下所示:
AService生命周期 1.實例化 --->AService的普通對象---->存入boziMap<beanName,AService的普通對象> 2.填充BService(添加了AutoWired)-->從單例池中獲取BService(此時單例池中可能還沒存放BService的Bean)-->創(chuàng)建BService 當(dāng)單例池中可能還沒存放BService的Bean時,觸發(fā)創(chuàng)建BService生命周期,步驟和AService一致。如下所示: 2.1.實例化--->BService的普通對象 2.2.填充AService(添加了AutoWired)-->單例池(此時AAService也在創(chuàng)建中,還沒放入單例池,如果創(chuàng)建的話,就會出現(xiàn)循環(huán)依賴)--->boziMap中取AService對象。 2.3.填充其他屬性 2.4.其他步驟(包括AOP) 2.5.加入到單例池中 3.填充其他屬性 4.其他步驟(包括AOP)--> AService代理對象(AServiceProxy) 5.加入到單例池中
其中AService對象的代理對象和AService的代理對象如下所示:
示例2中通過一個boziMap打破了循環(huán)創(chuàng)建bean對象,而產(chǎn)生的循環(huán)依賴。而在AOP過程中,步驟2.2.填充AService時,應(yīng)該是把4.其他步驟,AService代理對象賦值給步驟2.2.填充AService的普通對象。所以要對示例2進(jìn)行優(yōu)化,把AOP放到第二步,先判斷是否出現(xiàn)循環(huán)依賴,在進(jìn)行AOP,再把AService的代理對象放入biziMap中。得到示例3:
AService生命周期 0.creatingSet[AService]表示正在創(chuàng)建中的Bean。 1.實例化 --->AService的普通對象 2.填充BService(添加了AutoWired)-->從單例池中獲取BService (此時單例池中可能還沒存放BService的Bean)-->創(chuàng)建BService 當(dāng)單例池中可能還沒存放BService的Bean時,觸發(fā)創(chuàng)建BService生命周期,步驟和AService一致。如下所示: 2.1.實例化--->BService的普通對象 2.2.填充aService(添加了AutoWired)-->單例池-->creatingSet中是否存在AService-->AOP-->AService的代理對象-->最后賦值給BService中的屬性aService 2.3.填充其他屬性 2.4.其他步驟 2.5.加入到單例池中 3.填充其他屬性 4.其他步驟(包括AOP)--> AService代理對象(AServiceProxy) 5.加入到單例池中
此時加大難度,多加一個CService,C中依賴的A,而A中也依賴了C,根據(jù)示例3,會得到如下步驟。把本例子定義為示例4,該例子中,填充bService和cService會反復(fù)創(chuàng)建代理對象。
AService生命周期 0.creatingSet[AService]表示正在創(chuàng)建中的Bean。 1.實例化 --->AService的普通對象 2.填充BService,CService(添加了AutoWired)-->從單例池中獲取BService (此時單例池中可能還沒存放BService的Bean)-->創(chuàng)建BService 當(dāng)單例池中可能還沒存放BService的Bean時,觸發(fā)創(chuàng)建BService生命周期,步驟和AService一致。如下所示: 2.1.實例化--->BService的普通對象 2.2.填充aService(添加了AutoWired)-->單例池-->creatingSet中是否存在AService-->AOP-->AService的代理對象-->最后賦值給BService中的屬性aService 2.3.填充其他屬性 2.4.其他步驟 2.5.加入到單例池中 填充CService,CService的生命周期 2.1.實例化-->BService的普通對象 2.2.填充aService(添加了AutoWired)-->creatingSet中是否存在AService-->AOP-->AService的代理對象-->最后賦值給CService中的屬性aService 2.3.填充其他屬性 2.4.其他步驟 2.5.加入到單例池中 3.填充其他屬性 4.其他步驟(包括AOP)--> AService代理對象(AServiceProxy) 5.加入到單例池中
為了解決這個問題,使用二級緩存存放A的代理對象。
AService生命周期 0.creatingSet[AService]表示正在創(chuàng)建中的Bean。 1.實例化 --->AService的普通對象 2.填充BService,CService(添加了AutoWired)-->從單例池中獲取BService (此時單例池中可能還沒存放BService的Bean)-->創(chuàng)建BService 當(dāng)單例池中可能還沒存放BService的Bean時,觸發(fā)創(chuàng)建BService生命周期,步驟和AService一致。如下所示: 2.1.實例化--->BService的普通對象 2.2.填充aService(添加了AutoWired)-->單例池-->creatingSet中是否存在AService-->循環(huán)依賴 --->earlySingletonObjects是否存在aService的代理對象,存在返回bean即可 -->AOP-->AService的代理對象-->存入二級緩存,earlySingletonObjects<beanName,AService的代理對象>-->最后賦值給BService中的屬性aService 2.3.填充其他屬性 2.4.其他步驟 2.5.加入到單例池中 填充CService,CService的生命周期 2.1.實例化-->BService的普通對象 2.2.填充aService(添加了AutoWired)-->單例池-->creatingSet中是否存在AService-->earlySingletonObjects 2.3.填充其他屬性 2.4.其他步驟 2.5.加入到單例池中 3.填充其他屬性 4.其他步驟(包括AOP)--> AService代理對象(AServiceProxy) 5.加入到單例池中
三級緩存
三級緩存為了打破循環(huán),在第一步驟中生成的不同對象,用三級緩存保存起來。三級緩存在Spring‘源碼中可singletonFactories,是一個Map,其中它的value存的是一個lambda表達(dá)式,其中的邏輯是,判斷是否需要AOP,需要則返回一個代理對象,反之則返回Service的普通對象。’
小結(jié)
一級緩存的作用就是保存經(jīng)過完整生命周期的Bean對象。
二級緩存,早期由于出現(xiàn)循環(huán)依賴,保存那些還沒完整走完bean生命周期的bean對象,是為了給其他Bean填充屬性時使用的代理對象賦值。
三級緩存,在出現(xiàn)循環(huán)依賴時,如果二級緩存中沒有存有對應(yīng)的bean對象,需要通過三級緩存去判斷,是否需要AOP,是則需要返回代理對象,否則需要返回普通對象。三級返回的的結(jié)果最終還是存在二級緩存中。Spring的核心源碼實現(xiàn)如下所示。
到此這篇關(guān)于Spring為什么要用三級緩存解決循環(huán)依賴的文章就介紹到這了,更多相關(guān)Spring三級緩存解決循環(huán)依賴內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring Boot 集成 RocketMQ 全流程指南(從依賴引入到消息收發(fā)
本文將通過 手動連接 和 配置連接 兩種方式,詳細(xì)講解如何在 Spring Boot 中集成 RocketMQ,實現(xiàn)消息的同步與異步發(fā)送,并提供完整示例代碼,感興趣的朋友一起看看吧2025-04-04Spring深入分析講解BeanUtils的實現(xiàn)
java知識體系統(tǒng)有很多數(shù)據(jù)實體,比較常用的DTO、BO、DO、VO等,其他類似POJO概念太老了現(xiàn)在基本廢棄掉了,本篇幅直接忽略,對于這幾種數(shù)據(jù)實體各自代表的含義和應(yīng)用場景先做一下簡單描述和分析2022-06-06MybatisPlus 插入或更新數(shù)據(jù)時自動填充更新數(shù)據(jù)解決方案
本文主要介紹了MybatisPlus 插入或更新數(shù)據(jù)時自動填充更新數(shù)據(jù)解決方案,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-09-09Spring+MyBatis實現(xiàn)數(shù)據(jù)讀寫分離的實例代碼
本篇文章主要介紹了Spring+MyBatis實現(xiàn)數(shù)據(jù)讀寫分離的實例代碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-07-07詳解jdbc實現(xiàn)對CLOB和BLOB數(shù)據(jù)類型的操作
這篇文章主要介紹了詳解jdbc實現(xiàn)對CLOB和BLOB數(shù)據(jù)類型的操作的相關(guān)資料,這里實現(xiàn)寫入操作與讀寫操作,需要的朋友可以參考下2017-08-08Java實現(xiàn)用位運算維護(hù)狀態(tài)碼
位運算是一種非常高效的運算方式,在算法考察中比較常見,那么業(yè)務(wù)代碼中我們?nèi)绾问褂梦贿\算呢,感興趣的小伙伴快跟隨小編一起學(xué)習(xí)一下吧2024-03-03