国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

Spring 動(dòng)態(tài)代理時(shí)是如何解決循環(huán)依賴的?為什么要使用三級(jí)緩存?

存儲(chǔ) 存儲(chǔ)軟件
在研究 『 Spring 是如何解決循環(huán)依賴的 』 的時(shí)候,了解到 Spring 是借助三級(jí)緩存來(lái)解決循環(huán)依賴的。

 [[379347]]

前言

在研究 『 Spring 是如何解決循環(huán)依賴的 』 的時(shí)候,了解到 Spring 是借助三級(jí)緩存來(lái)解決循環(huán)依賴的。

同樣在上一節(jié)留下了疑問(wèn):

  1. 循環(huán)依賴為什么要使用三級(jí)緩存?而不是使用二級(jí)緩存?
  2. AOP 動(dòng)態(tài)代理對(duì)循環(huán)依賴的有沒(méi)有什么影響?

本篇文章也是圍繞上面的內(nèi)容進(jìn)行展開(kāi)。

筆記也在不斷整理,之前可能會(huì)有點(diǎn)雜亂。

循序漸進(jìn),看一看什么是循環(huán)依賴?

先來(lái)回顧一下三級(jí)緩存的概念。

  • singletonObjects: 一級(jí)緩存,存儲(chǔ)單例對(duì)象,Bean 已經(jīng)實(shí)例化,初始化完成。
  • earlySingletonObjects: 二級(jí)緩存,存儲(chǔ) singletonObject,這個(gè) Bean 實(shí)例化了,還沒(méi)有初始化。
  • singletonFactories: 三級(jí)緩存,存儲(chǔ) singletonFactory。

Bean 的創(chuàng)建過(guò)程

  1. @Service 
  2. public class CircularServiceA { 
  3.     private String fieldA = "字段 A"

單例 Bean 的創(chuàng)建過(guò)程

 

通過(guò)上面的流程,可以看出 Spring 在創(chuàng)建 Bean 的過(guò)程中重點(diǎn)是在 AbstractAutowireCapableBeanFactory 中的以下三個(gè)步驟:

  1. 實(shí)例化 createBeanInstance: 其中實(shí)例化 Bean 并對(duì) Bean 進(jìn)行賦值,像例子中的 fieldA 字段在這里就會(huì)賦值。
  2. 屬性注入 populateBean: 可以理解為對(duì) Bean 里面的屬性進(jìn)行賦值。(會(huì)依賴其他 Bean)
  3. 初始化 initializeBean: 執(zhí)行初始化和 Bean 的后置處理器。

實(shí)例化賦值源碼可以閱讀:

BeanUtils.instantiateClass(constructorToUse)

如果要依賴其他 Bean 呢?

那如果 CircularServiceA 依賴了其他 Bean 呢?

A 依賴了 B

 

當(dāng) A 依賴了 B 的時(shí)候,在 createBeanInstance 這一步,并不會(huì)對(duì) B 進(jìn)行屬性賦值。

而是在 populatedBean 這里查找依賴項(xiàng),并創(chuàng)建 B。

循環(huán)依賴下的創(chuàng)建過(guò)程

循環(huán)依賴的場(chǎng)景,在上一篇文章已經(jīng)有所講解,這里僅僅畫(huà)圖說(shuō)明一下。

  1. @Service 
  2. public class CircularServiceA { 
  3.  
  4.     private String fieldA = "字段 A"
  5.  
  6.     @Autowired 
  7.     private CircularServiceB circularServiceB; 
  8.  
  9. @Service 
  10. public class CircularServiceB { 
  11.     @Autowired 
  12.     private CircularServiceA circularServiceA; 

 

 


A B 循環(huán)依賴

 

 

在 A 和 B 循環(huán)依賴的場(chǎng)景中:

B populatedBean 查找依賴項(xiàng) A 的時(shí)候,從一級(jí)緩存中雖然未獲取到 A,但是發(fā)現(xiàn) A 在創(chuàng)建中。

此時(shí),從三級(jí)緩存中獲取 A 的 singletonFactory 調(diào)用工廠方法,創(chuàng)建 getEarlyBeanReference A 的早期引用并返回。

B 引用到 A ,B 就可以初始化完畢,然后 A 同樣也可以初始化完畢了。

二級(jí)緩存能否解決循環(huán)依賴

通過(guò)上面的圖,仔細(xì)分析一下,其實(shí)把二級(jí)緩存拿掉,在 B 嘗試獲取 A 的時(shí)候直接返回 A 的實(shí)例,是不是也是可以的?

答案是:可以的!

但是為什么還是用三級(jí)緩存呢?

網(wǎng)上的很多資料說(shuō)是和動(dòng)態(tài)代理有關(guān)系,那就從動(dòng)態(tài)代理的方面繼續(xù)往下分析分析。

動(dòng)態(tài)代理的場(chǎng)景

在 JavaConfig(配置類(lèi)) 上添加 @EnableAspectJAutoProxy 注解,開(kāi)啟 AOP ,通過(guò) Debug 循序漸進(jìn)看一看動(dòng)態(tài)代理對(duì)循環(huán)依賴的影響。

動(dòng)態(tài)代理下,Bean 的創(chuàng)建過(guò)程

  1. @Service 
  2. public class CircularServiceA { 
  3.     private String fieldA = "字段 A"
  4.  
  5.     public void methodA() { 
  6.  
  7.         System.out.println("方法 A 執(zhí)行"); 
  8.  
  9.     } 
  10. @Aspect 
  11. @Component 
  12. public class AspectA { 
  13.  
  14.     @Before("execution(public void com.liuzhihang.circular.CircularServiceA.methodA())"
  15.     public void beforeA() { 
  16.         System.out.println("beforeA 執(zhí)行"); 
  17.     } 

只有 A 的情況下,給 A 添加切面,開(kāi)始 Debug。

前面的流程都相同,在 initializeBean 開(kāi)始出現(xiàn)差異。

這一步需要初始化 Bean 并執(zhí)行 Bean 的后置處理器。

 


執(zhí)行后置處理器

 

 

其中有一個(gè)處理器為:AnnotationAwareAspectJAutoProxyCreator 其實(shí)就是加的注解切面,會(huì)跳轉(zhuǎn)到 AbstractAutoProxyCreator 類(lèi)的 postProcessAfterInitialization 方法

 

 


postProcessAfterInitialization

 

 

如圖所示:wrapIfNecessary 方法會(huì)判斷是否滿足代理?xiàng)l件,是的話返回一個(gè)代理對(duì)象,否則返回當(dāng)前 Bean。

后續(xù)調(diào)用 getProxy 、createAopProxy 等等,最終執(zhí)行到下面一部分。

 

最終會(huì)執(zhí)行到這里,AOP 代理相關(guān)的就不細(xì)看了。

一路放行,直到 initializeBean 執(zhí)行結(jié)束。

A 被替換為了代理對(duì)象

 

此時(shí)發(fā)現(xiàn):A 被替換為了代理對(duì)象。

所以 doCreateBean 返回,以及后面放到一級(jí)緩存中的都是代理對(duì)象。

紅框部分為差異

 

有循環(huán)依賴的動(dòng)態(tài)代理

這一次把循環(huán)依賴打開(kāi):

  1. @Service 
  2. public class CircularServiceA { 
  3.  
  4.     private String fieldA = "字段 A"
  5.  
  6.     @Autowired 
  7.     private CircularServiceB circularServiceB; 
  8.  
  9.     public void methodA() { 
  10.  
  11.         System.out.println("方法 A 執(zhí)行"); 
  12.  
  13.     } 
  14. @Aspect 
  15. @Component 
  16. public class AspectA { 
  17.  
  18.     @Before("execution(public void com.liuzhihang.circular.CircularServiceA.methodA())"
  19.     public void beforeA() { 
  20.  
  21.         System.out.println("beforeA 執(zhí)行"); 
  22.  
  23.     } 
  24.  
  25. @Service 
  26. public class CircularServiceB { 
  27.  
  28.     @Autowired 
  29.     private CircularServiceA circularServiceA; 
  30.  
  31.     public void methodB() { 
  32.  
  33.     } 
  34. @Aspect 
  35. @Component 
  36. public class AspectB { 
  37.      
  38.     @Before("execution(public void com.liuzhihang.circular.CircularServiceB.methodB())"
  39.     public void beforeB() { 
  40.  
  41.         System.out.println("beforeB 執(zhí)行"); 
  42.  
  43.     } 
  44.  

開(kāi)始 Debug,前面的一些列流程,都和正常的沒(méi)有什么區(qū)別。而唯一的區(qū)別在于,創(chuàng)建 B 的時(shí)候,需要從三級(jí)緩存獲取 A。

此時(shí)在 getSingleton 方法中會(huì)調(diào)用:singletonObject = singletonFactory.getObject();

B 屬性賦值時(shí),從三級(jí)緩存獲取 A

 

有時(shí)會(huì)比較疑惑 singletonFactory.getObject() 調(diào)用的是哪里?

 


三級(jí)緩存獲取對(duì)象

 

 

所以這一塊調(diào)用的是 getEarlyBeanReference,開(kāi)始遍歷執(zhí)行 BeanPostProcessor。

getEarlyBeanReference

 

 


getEarlyBeanReference

 

 

看到 wrapIfNecessary 就明白了吧!這塊會(huì)獲取一個(gè)代理對(duì)象。

也就是說(shuō)此時(shí)返回,并放到二級(jí)緩存的是一個(gè) A 的代理對(duì)象。

這樣 B 就創(chuàng)建完畢了!

到 A 開(kāi)始初始化并執(zhí)行后置處理器了!因?yàn)?A 也有代理,所以 A 也會(huì)執(zhí)行到 postProcessAfterInitialization 這一部分!

 


判斷二級(jí)緩存

 

 

但是在執(zhí)行 wrapIfNecessary 之前,會(huì)先判斷二級(jí)緩存是否有 A 了。

  1. this.earlyProxyReferences.remove(cacheKey) != bean 

但是這塊獲取到的是 A 的代理對(duì)象。肯定是 false 。所以不會(huì)再生成一次 A 的代理對(duì)象。

代理 - 循環(huán)依賴

 

總結(jié)

可以看到,循環(huán)依賴下,有沒(méi)有代理情況下的區(qū)別就在:

  1. singletonObject = singletonFactory.getObject(); 

在循環(huán)依賴發(fā)生的情況下 B 中的 A 賦值時(shí):

  1. 無(wú)代理:getObject 直接返回原來(lái)的 Bean
  2. 有代理:getObject 返回的是代理對(duì)象

然后都放到二級(jí)緩存。

為什么要三級(jí)緩存?

假設(shè)去掉三級(jí)緩存

去掉三級(jí)緩存之后,Bean 直接創(chuàng)建 earlySingletonObjects, 看著好像也可以。

如果有代理的時(shí)候,在 earlySingletonObjects 直接放代理對(duì)象就行了。

但是會(huì)導(dǎo)致一個(gè)問(wèn)題:在實(shí)例化階段就得執(zhí)行后置處理器,判斷有 AnnotationAwareAspectJAutoProxyCreator 并創(chuàng)建代理對(duì)象。

這么一想,是不是會(huì)對(duì) Bean 的生命周期有影響。

同樣,先創(chuàng)建 singletonFactory 的好處就是:在真正需要實(shí)例化的時(shí)候,再使用 singletonFactory.getObject() 獲取 Bean 或者 Bean 的代理。相當(dāng)于是延遲實(shí)例化。

假設(shè)去掉二級(jí)緩存

如果去掉了二級(jí)緩存,則需要直接在 singletonFactory.getObject() 階段初始化完畢,并放到一級(jí)緩存中。

 


B 和 C 都依賴 A

 

 

那有這么一種場(chǎng)景,B 和 C 都依賴了 A。

要知道在有代理的情況下 singletonFactory.getObject() 獲取的是代理對(duì)象。

 

 


多次獲取代理對(duì)象不同

 

 

而多次調(diào)用 singletonFactory.getObject() 返回的代理對(duì)象是不同的,就會(huì)導(dǎo)致 B 和 C 依賴了不同的 A。

那如果獲取 B 到之后直接放到一級(jí)緩存,然后 C 再獲取呢?

 

一級(jí)緩存放的是已經(jīng)初始化完畢的 Bean,要知道 A 依賴了 B 和 C ,A 這時(shí)候還沒(méi)有初始化完畢。

小結(jié)

循環(huán)依賴的場(chǎng)景有很多,本文只是通過(guò) Debug ,來(lái)了解到循環(huán)依賴和 AOP 之間的關(guān)系,以及了解到為什么要用三級(jí)緩存。

當(dāng)然,Spring 設(shè)計(jì)之初是什么樣子的?如何一步一步發(fā)展成現(xiàn)在這種的?

肯定是不能慢慢去研究了,所以只能以現(xiàn)在的版本,去揣測(cè)作者的意圖。

不足之處,多多指正。

本文轉(zhuǎn)載自微信公眾號(hào)「程序員小航」,可以通過(guò)以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系程序員小航公眾號(hào)。

 

責(zé)任編輯:武曉燕 來(lái)源: 程序員小航
相關(guān)推薦

2023-12-12 17:44:13

三級(jí)緩存Bean

2023-02-26 11:15:42

緩存循環(huán)依賴

2022-05-08 19:23:28

Spring循環(huán)依賴

2020-12-29 08:34:08

spring循環(huán)依賴開(kāi)發(fā)

2022-12-02 12:01:30

Spring緩存生命周期

2022-03-01 18:03:06

Spring緩存循環(huán)依賴

2024-03-04 08:47:17

Spring框架AOP

2019-11-26 14:30:20

Spring循環(huán)依賴Java

2023-11-28 08:00:00

SpringJava

2025-06-26 01:55:00

2023-10-07 08:40:57

緩存屬性Spring

2025-07-02 03:10:00

2022-08-17 07:52:31

Spring循環(huán)依賴單例池

2024-12-20 16:46:22

Spring三級(jí)緩存

2024-04-15 08:17:21

Spring依賴注入循環(huán)依賴

2023-08-09 10:43:21

源碼循環(huán)依賴getBean

2020-06-22 08:07:48

Spring依賴場(chǎng)景

2015-09-22 11:09:47

Java 8動(dòng)態(tài)代理

2023-09-21 09:00:00

Merge Que開(kāi)發(fā)工具Mergify

2020-11-27 06:28:55

Spring循環(huán)依賴
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

国产精品77777| 91精品国产综合久久福利软件| 亚洲影影院av| 巨大荫蒂视频欧美大片| 国产成人久久精品77777最新版本| 91国产美女视频| 国产丝袜视频在线播放| 中文字幕中文字幕一区| 视频三区二区一区| 91精品综合久久久久久久久久久 | 欧美日韩国产小视频在线观看| r级无码视频在线观看| 中文字幕免费一区二区| 欧美高清第一页| 欧美xxxx黑人又粗又长| 亚洲国产欧美在线| 日本男人操女人| 国产一区二区久久| 国精产品99永久一区一区| 欧美一区二区三区久久| 一区二区在线视频播放| 色爱综合区网| 欧美三区免费完整视频在线观看| 成人黄色电影在线| 成人精品在线视频观看| 伊人色综合久久天天五月婷| 黄色成人在线网站| 国产精品爽黄69| 日韩mv欧美mv国产网站| 欧美成人一二三| 欧美色网在线| 亚洲第一网站男人都懂| av免费在线一区二区三区| 亚洲高清中文字幕| 成人免费视频网站在线看| 成人免费视频一区二区| 国产又黄又爽免费视频| 久久午夜激情| 欧美精品在线一区| 欧美视频二区| 91免费看国产| 中文无码久久精品| 91美女片黄在线观| 欧美性感美女一区二区| 日本精品免费一区二区三区| 天堂综合网久久| 欧美在线www| 外国成人在线视频| 777午夜精品福利在线观看| 国产精品传媒| 91大神在线播放精品| 国产乱人伦精品一区| 午夜精品国产精品大乳美女| 成人三级av在线| 8090成年在线看片午夜| 91精品啪在线观看国产爱臀 | 欧美日韩大陆在线| 午夜视频在线免费观看| 日韩一区二区精品| 国产精品论坛| 亚洲图片在区色| 欧美日韩在线视频首页| 日韩欧美亚洲| 亚洲久色影视| 国产精品一区二区欧美黑人喷潮水| 91午夜在线| 日韩高清a**址| 色999久久久精品人人澡69| www.日本在线播放| 中文字幕欧美日韩精品| 九九九九九九精品任你躁| 国自产拍偷拍精品啪啪一区二区| 精品呦交小u女在线| 成人免费福利片| 免费av高清| 亚洲国产精品成人精品| 欧美日韩高清| 国产精品久久久久久亚洲影视| 久久久精品日韩欧美| 男人在线资源站| 最好看的2019年中文视频| 麻豆精品99| 国产综合18久久久久久| 日韩精品免费在线视频观看| 床上的激情91.| 91桃色在线| 精品久久久噜噜噜噜久久图片| 精品88久久久久88久久久| 全国精品免费看| 国产在线视频你懂得| 91文字幕巨乱亚洲香蕉| 亚洲免费电影在线观看| 久久蜜臀中文字幕| 性xxxx欧美老肥妇牲乱| 日本在线视频网| 国产精品二区在线| 国产不卡在线播放| 亚洲青青一区| 免费一级淫片aaa片毛片a级| 欧美一级淫片丝袜脚交| 亚洲精品伦理在线| 福利在线视频网站| 日韩亚洲视频在线| 亚洲国产成人在线视频| 国内精品自线一区二区三区视频| cao在线视频| 亚洲乱码日产精品bd在线观看| 国产91色在线|| 在线不卡一区二区| 久久国产66| 色综合咪咪久久网| 九色porny丨首页入口在线| 亚洲欧美视频二区| 欧美少妇一区二区| 精品91福利视频| 国产一区二区三区黄| 久久色.com| 中文字幕在线观| 亚洲精品永久免费精品| 国产精品不卡| 天堂av在线网站| 精品无人区太爽高潮在线播放| 中国av一区| 女人被男人躁得好爽免费视频| 日韩欧美精品在线观看| 欧美h版在线观看| 久久国产日韩欧美| 樱桃视频在线观看一区| 久久天天久久| 日韩欧美在线观看强乱免费| 亚洲高清免费在线| 国产精品jk白丝蜜臀av小说 | 伊人网在线免费| 日韩欧美综合在线视频| 97品白浆高清久久久久久| 自拍视频一区二区三区| 激情成人在线视频| 日韩在线观看一区二区三区| 亚洲精品二区| 欧美私人免费视频| 色999国产精品| 国产免费福利| 欧美国产日韩视频| 国产精品一色哟哟哟| dy888亚洲精品一区二区三区| 国产精品旅馆在线| 亚洲欧洲国产日韩| 99久热这里只有精品视频免费观看| 国产av第一区| 亚洲精品久久久久久下一站| 国产精品久久久久久模特| 青青草手机在线| 91免费高清视频| 欧美日韩在线视频一区| 国产精品视频你懂的| 成人一区视频| 91免费国产精品| 亚洲国产天堂网精品网站| 丝袜诱惑制服诱惑色一区在线观看 | 国产乱妇无码大片在线观看| 欧美成人精品一区二区男人看| 国产综合色香蕉精品| 亚洲综合在线免费观看| 最新亚洲精品| 性史性dvd影片农村毛片| 国产精品久久久久久久久久久不卡| 亚洲欧美另类图片小说| 精品欧美久久| 在线看的网站你懂| 91精品国产综合久久久久久丝袜 | 一级免费视频| 欧洲亚洲女同hd| 天海翼一区二区三区免费| 欧美国产乱视频| 中文字幕av一区二区三区高| av不卡一区| y4480在线8影院| 国产精品乱子乱xxxx| 日韩一区二区免费在线电影| 美女免费视频一区二区| 色综合天天色| mm131亚洲精品| 国产99久久精品一区二区永久免费| 一区二区激情小说| 亚洲第一区色| 中文字幕在线中文字幕在线中三区| 国产成a人亚洲精v品在线观看| 久久久99免费视频| 亚洲综合色区另类av| 亚洲精选国产| se01亚洲视频| 99视频资源网| 国产一区二区视频在线免费观看| 欧美电视剧在线看免费| 国产成人精品亚洲午夜麻豆| 日本久久伊人| 欧美777四色影视在线| 伊人久久婷婷色综合98网| 欧美激情第1页| 欧美影院一区二区|