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