成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術(shù)文章
文章詳情頁

spring 如何解決循環(huán)依賴

瀏覽:118日期:2023-07-22 15:07:07

首先解釋下什么是循環(huán)依賴,其實(shí)很簡單,就是有兩個類它們互相都依賴了對方,如下所示:

@Componentpublic class AService { @Autowired private BService bService;}

@Componentpublic class BService { @Autowired private AService aService;}

AService和BService顯然兩者都在內(nèi)部依賴了對方,單拎出來看仿佛看到了多線程中常見的死鎖代碼,但很顯然Spring解決了這個問題,不然我們也不可能正常的使用它了。

所謂創(chuàng)建Bean實(shí)際上就是調(diào)用getBean() 方法,這個方法可以在AbstractBeanFactory這個類里面找到,這個方法一開始會調(diào)用getSingleton()方法。

// Eagerly check singleton cache for manually registered singletons.Object sharedInstance = getSingleton(beanName);

這個方法的實(shí)現(xiàn)長得很有意思,有著一堆if語句。

protected Object getSingleton(String beanName, boolean allowEarlyReference) { Object singletonObject = this.singletonObjects.get(beanName); if (singletonObject == null && this.isSingletonCurrentlyInCreation(beanName)) { singletonObject = this.earlySingletonObjects.get(beanName); if (singletonObject == null && allowEarlyReference) { synchronized(this.singletonObjects) {singletonObject = this.singletonObjects.get(beanName);if (singletonObject == null) { singletonObject = this.earlySingletonObjects.get(beanName); if (singletonObject == null) { ObjectFactory<?> singletonFactory = (ObjectFactory)this.singletonFactories.get(beanName); if (singletonFactory != null) { singletonObject = singletonFactory.getObject(); // 從三級緩存里取出放到二級緩存中 this.earlySingletonObjects.put(beanName, singletonObject); this.singletonFactories.remove(beanName); } }} } } } return singletonObject;}

但這一坨if很好理解,就是一層層的去獲取這個bean,首先從singletonObjects中獲取,這里面存放的是已經(jīng)完全創(chuàng)建好的單例Bean;如果取不到,那么就往下走,去earlySingletonObjects里面取,這個是早期曝光的對象;如果還是沒有,那么再去第三級緩存singletonFactories里面獲取,它是提前暴露的對象工廠,這里會從三級緩存里取出后放到二級緩存中。那么總的來說,Spring去獲取一個bean的時候,其實(shí)并不是直接就從容器里面取,而是先從緩存里找,而且緩存一共有三級。那么從這個方法返回的并不一定是我們需要的bean,后面會調(diào)用getObjectForBeanInstance()方法去得到實(shí)例化后的bean,這里就不多說了。

但如果緩存里面的確是取不到bean呢?那么說明這個bean的確還未創(chuàng)建,需要去創(chuàng)建一個bean,這樣我們就會去到前一篇生命周期中的創(chuàng)建bean的方法了?;仡櫹铝鞒蹋簩?shí)例化?屬性注入?初始化?銷毀。那么我們回到文章開頭的例子,有ServiceA和ServiceB兩個類。一般來說,Spring是按照自然順序去創(chuàng)建bean,那么第一個要創(chuàng)建的是ServiceA。顯然一開始緩存里是沒有的,我們會來到創(chuàng)建bean的方法。首先進(jìn)行實(shí)例化階段,我們會來到第一個跟解決循環(huán)依賴有關(guān)的代碼,在實(shí)例化階段的代碼中就可以找到。

// 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.isTraceEnabled()) { logger.trace('Eagerly caching bean ’' + beanName + '’ to allow for resolving potential circular references'); } addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));}

首先看看第一行,earlySingletonExposure這個變量它會是什么值?

它是有一個條件表達(dá)式返回的,一個個來看,首先,mbd.isSingleton()。我們知道Spring默認(rèn)的Bean的作用域都是單例的,因此這里正常來說都是返回true沒問題。第二個,this.allowCircularReference,這個變量是標(biāo)記是否允許循環(huán)引用,默認(rèn)也是true。第三個,調(diào)用了一個方法,isSingletonCurrentlyInCreation(beanName),進(jìn)入該代碼可以看出它是返回當(dāng)前的bean是不是正常創(chuàng)建,顯然也是true。因此這個earlySingletonExposure返回的就是true。

接下來就進(jìn)入了if語句的實(shí)現(xiàn)里面了,也就是addSingletonFactory()這個方法??吹嚼锩娴拇a中出現(xiàn)singletonFactories這個變量是不是很熟悉?翻到上面的getSingleton()就知道了,其實(shí)就是三級緩存,所以這個方法的作用是通過三級緩存提前暴露一個工廠對象。

/** * 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é)說的實(shí)例化之后的步驟,就是屬性注入了。這就意味著ServiceA需要將ServiceB注入進(jìn)去,那么顯然又要調(diào)用getBean()方法去獲取ServiceB。ServiceB還沒有創(chuàng)建,則也會進(jìn)入這個createBean()方法,同樣也會來到這一步依賴注入。ServiceB中依賴了ServiceA,則會調(diào)用getBean()去獲取ServiceA。此時的獲取ServiceA可就不是再創(chuàng)建Bean了,而是從緩存中獲取。這個緩存就是上面getSingleton()這個方法里面我們看到的singletonFactory。那么這個singletonFactory哪里來的,就是這個addSingletonFactory()方法的第二個參數(shù),即getEarlyBeanReference()方法。

/** * Obtain a reference for early access to the specified bean, * typically for the purpose of resolving a circular reference. * @param beanName the name of the bean (for error handling purposes) * @param mbd the merged bean definition for the bean * @param bean the raw bean instance * @return the object to expose as bean reference */protected Object getEarlyBeanReference(String beanName, RootBeanDefinition mbd, Object bean) { Object exposedObject = bean; if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) { for (SmartInstantiationAwareBeanPostProcessor bp : getBeanPostProcessorCache().smartInstantiationAware) { exposedObject = bp.getEarlyBeanReference(exposedObject, beanName); } } return exposedObject;}

查看bp.getEarlyBeanReference(exposedObject, beanName)的實(shí)現(xiàn),發(fā)現(xiàn)有兩個,一個是spring-beans下的SmartInstantiationAwareBeanPostProcessor,一個是spring-aop下的AbstractAutoProxyCreator。我們在未使用AOP的情況下,取的還是第一種實(shí)現(xiàn)。

default Object getEarlyBeanReference(Object bean, String beanName) throws BeansException { return bean;}

那么令人驚訝的是,這方法直接返回了bean,也就是說如果不考慮AOP的話,這個方法啥都沒干,就是把實(shí)例化創(chuàng)建的對象直接返回了。如果考慮AOP的話調(diào)用的是另一個實(shí)現(xiàn):

public Object getEarlyBeanReference(Object bean, String beanName) { Object cacheKey = getCacheKey(bean.getClass(), beanName); this.earlyProxyReferences.put(cacheKey, bean); return wrapIfNecessary(bean, beanName, cacheKey);}

可以看出,如果使用了AOP的話,這個方法返回的實(shí)際上是bean的代理,并不是它本身。那么通過這部分我們可以認(rèn)為,在沒有使用AOP的情況下,三級緩存是沒有什么用的,所謂三級緩存實(shí)際上只是跟Spring的AOP有關(guān)的。

好了我們現(xiàn)在是處于創(chuàng)建B的過程,但由于B依賴A,所以調(diào)用了獲取A的方法,則A從三級緩存進(jìn)入了二級緩存,得到了A的代理對象。當(dāng)然我們不需要擔(dān)心注入B的是A的代理對象會帶來什么問題,因?yàn)樯纱眍惖膬?nèi)部都是持有一個目標(biāo)類的引用,當(dāng)調(diào)用代理對象的方法的時候,實(shí)際上是會調(diào)用目標(biāo)對象的方法的,所以所以代理對象是沒影響的。當(dāng)然這里也反應(yīng)了我們實(shí)際上從容器中要獲取的對象實(shí)際上是代理對象而不是其本身。

那么我們再回到創(chuàng)建A的邏輯往下走,能看到后面實(shí)際上又調(diào)用了一次getSingleton()方法。傳入的allowEarlyReference為false。

if (earlySingletonExposure) { Object earlySingletonReference = getSingleton(beanName, false); if (earlySingletonReference != null) { if (exposedObject == bean) { exposedObject = earlySingletonReference; } ... }}

翻看上面的getSingleton()代碼可以看出,allowEarlyReference為false就相當(dāng)于禁用三級緩存,代碼只會執(zhí)行到通過二級緩存get。

singletonObject = this.earlySingletonObjects.get(beanName);

因?yàn)樵谇懊嫖覀冊趧?chuàng)建往B中注入A的時候已經(jīng)從三級緩存取出來放到二級緩存中了,所以這里A可以通過二級緩存去取。再往下就是生命周期后面的代碼了,就不再繼續(xù)了。

那么現(xiàn)在就會有個疑問,我們?yōu)槭裁捶且壘彺?,直接用二級緩存似乎就足夠了?/p>

看看上面getEarlyBeanReference()這個方法所在的類,它是SpringAOP自動代理的關(guān)鍵類,它實(shí)現(xiàn)了SmartInstantiationAwareBeanPostProcessor,也就是說它也是個后置處理器BeanPostProcessor,它有著自定義的初始化后的方法。

/** * Create a proxy with the configured interceptors if the bean is * identified as one to proxy by the subclass. * @see #getAdvicesAndAdvisorsForBean */@Overridepublic Object postProcessAfterInitialization(@Nullable Object bean, String beanName) { if (bean != null) { Object cacheKey = getCacheKey(bean.getClass(), beanName); if (this.earlyProxyReferences.remove(cacheKey) != bean) { return wrapIfNecessary(bean, beanName, cacheKey); } } return bean;}

很明顯它這里是earlyProxyReferences緩存中找不到當(dāng)前的bean的話就會去創(chuàng)建代理。也就是說SpringAOP希望在Bean初始化后進(jìn)行創(chuàng)建代理。如果我們只使用二級緩存,也就是在這個地方

addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));

直接調(diào)用getEarlyBeanReference()并將得到的早期引用放入二級緩存。這就意味著無論bean之間是否存在互相依賴,只要創(chuàng)建bean走到這一步都得去創(chuàng)建代理對象了。然而Spring并不想這么做,不信自己可以動手debug一下,如果ServiceA和ServiceB之間沒有依賴關(guān)系的話,getEarlyBeanReference()這個方法壓根就不會執(zhí)行??偟膩碚f就是,如果不使用三級緩存直接使用二級緩存的話,會導(dǎo)致所有的Bean在實(shí)例化后就要完成AOP代理,這是沒有必要的。

最后我們重新梳理下流程,記得Spring創(chuàng)建Bean的時候是按照自然順序的,所以A在前B在后:

spring 如何解決循環(huán)依賴

我們首先進(jìn)行A的創(chuàng)建,但由于依賴了B,所以開始創(chuàng)建B,同樣的,對B進(jìn)行屬性注入的時候會要用到A,那么就會通過getBean()去獲取A,A在實(shí)例化階段會提前將對象放入三級緩存中,如果沒有使用AOP,那么本質(zhì)上就是這個bean本身,否則是AOP代理后的代理對象。三級緩存singletonFactories會將其存放進(jìn)去。那么通過getBean()方法獲取A的時候,核心其實(shí)在于getSingleton()方法, 它會將其從三級緩存中取出,然后放到二級緩存中去。而最終B創(chuàng)建結(jié)束回到A初始化的時候,會再次調(diào)用一次getSingleton()方法,此時入?yún)⒌腶llowEarlyReference為false,因此是去二級緩存中取,得到真正需要的bean或代理對象,最后A創(chuàng)建結(jié)束,流程結(jié)束。

所以Spring解決循環(huán)依賴的原理大致就講完了,但根據(jù)上述的結(jié)論,我們可以思考一個問題,什么情況的循環(huán)依賴是無法解決的?

根據(jù)上面的流程圖,我們知道,要解決循環(huán)依賴首先一個大前提是bean必須是單例的,基于這個前提我們才值得繼續(xù)討論這個問題。然后根據(jù)上述總結(jié),可以知道,每個bean都是要進(jìn)行實(shí)例化的,也就是要執(zhí)行構(gòu)造器。所以能不能解決循環(huán)依賴問題其實(shí)跟依賴注入的方式有關(guān)。

依賴注入的方式有setter注入,構(gòu)造器注入和Field方式。

Filed方式就是我們平時用的最多的,屬性上加個@Autowired或者@Resource之類的注解,這個對解決循環(huán)依賴無影響;

如果A和B都是通過setter注入,顯然對于執(zhí)行構(gòu)造器沒有影響,所以不影響解決循環(huán)依賴;

如果A和B互相通過構(gòu)造器注入,那么執(zhí)行構(gòu)造器的時候也就是實(shí)例化的時候,A在自己還沒放入緩存的時候就去創(chuàng)建B了,那么B也是拿不到A的,因此會出錯;

如果A中注入B的方式為setter,B中注入A為構(gòu)造器,由于A先實(shí)例化,執(zhí)行構(gòu)造器,并創(chuàng)建緩存,都沒有問題,繼續(xù)屬性注入,依賴了B然后走創(chuàng)建B的流程,獲取A也可以從緩存里面能取到,流程一路通暢。

如果A中注入B的方式為構(gòu)造器,B中注入A為setter,那么這個時候A先進(jìn)入實(shí)例化方法,發(fā)現(xiàn)需要B,那么就會去創(chuàng)建B,而A還沒放入三級緩存里,B再創(chuàng)建的時候去獲取A就會獲取失敗。

好了,以上就是關(guān)于Spring解決循環(huán)依賴問題的所有內(nèi)容,這個問題的答案我是很久之前就知道了,但真的只是知道答案,這次是自己看源碼加debug一點(diǎn)點(diǎn)看才知道為啥是這個答案,雖然還做不到徹底學(xué)的通透,但的確能對這個問題的理解的更為深刻一點(diǎn),再接再厲吧。

以上就是spring 如何解決循環(huán)依賴的詳細(xì)內(nèi)容,更多關(guān)于spring 循環(huán)依賴的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Spring
相關(guān)文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
久久69国产一区二区蜜臀 | 理论片日本一区| 国产精品欧美久久| 日韩欧美二区三区| 国产成人精品免费在线| 欧美三区在线观看| 日韩av一级电影| 亚洲精品自在在线观看| 日韩美女精品在线| 亚洲国产一区二区三区a毛片| 欧美高清在线一区二区| 欧美日韩p片| 国产拍欧美日韩视频二区| 你懂的一区二区| 久久久国产精品麻豆| 99久久久无码国产精品| 精品美女一区二区| 成人精品国产一区二区4080| 欧美一区二区三区播放老司机| 国产尤物一区二区| 欧美一区三区二区| 懂色av中文字幕一区二区三区| 91精品国产综合久久精品图片 | 国产模特精品视频久久久久| 亚洲久本草在线中文字幕| 亚洲精品一区二区三区樱花| 亚洲日本在线看| 国产欧美丝祙| 一区二区在线观看不卡| 国产欧美日韩综合一区在线播放| 一区二区三区视频在线看| 国产精品亚洲综合色区韩国| 石原莉奈一区二区三区在线观看| 色一情一伦一子一伦一区| 偷拍自拍另类欧美| 欧美日韩激情一区二区| 国产凹凸在线观看一区二区| 久久亚洲精品国产精品紫薇| 国产精品v欧美精品∨日韩| 亚洲摸摸操操av| 亚洲免费影院| 蜜臀a∨国产成人精品| 777奇米四色成人影色区| 丁香婷婷深情五月亚洲| 国产日产欧美一区二区视频| 日韩视频精品在线观看| 日韩精品久久理论片| 69精品人人人人| 欧美在线网址| 亚洲欧美日韩一区| 色狠狠综合天天综合综合| 国产精品一区一区三区| 国产午夜精品福利| 国产精品一区二区a| 久久99精品国产.久久久久| 精品久久久三级丝袜| 免费看精品久久片| 午夜久久久影院| 久久中文字幕一区二区三区| 久久国产精品99精品国产| 欧美一级搡bbbb搡bbbb| 欧美不卡三区| 一区二区三区在线播| 蜜桃久久精品乱码一区二区| 精品一区二区综合| 精品国产乱子伦一区| 精品999网站| 日韩精品乱码免费| 精品免费一区二区三区| 亚洲福利av| 青青草97国产精品免费观看| 欧美大片一区二区| 在线精品福利| 免费观看在线综合| 欧美精品一区二区三区四区| 亚洲高清在线观看一区| 美女精品一区二区| www国产精品av| 国产精品外国| 国产99久久精品| 亚洲美女一区二区三区| 色999日韩国产欧美一区二区| 国产91丝袜在线播放九色| 国产精品不卡一区二区三区| 色94色欧美sute亚洲13| 99久久夜色精品国产网站| 亚洲激情图片qvod| 4438成人网| 99精品国产高清一区二区| 国产在线播放一区三区四| 国产精品乱码妇女bbbb| 91官网在线观看| 欧美福利网址| 日韩电影一区二区三区四区| 26uuu精品一区二区三区四区在线| 在线天堂一区av电影| 国产九色sp调教91| 亚洲美女在线一区| 日韩欧美三级在线| 欧美一级专区| 91看片淫黄大片一级在线观看| 同产精品九九九| 久久亚洲一级片| 在线一区二区三区| 亚洲视频欧美在线| 国产精品1区2区| 亚洲国产精品久久不卡毛片| 久久久精品综合| 日本国产一区二区| 影音先锋亚洲一区| 国产精品一区三区| 亚洲一区二区三区四区在线| 久久一区二区三区国产精品| 在线观看国产日韩| 91久久国产综合久久蜜月精品| 高清不卡在线观看| 日本视频在线一区| 日韩美女精品在线| 精品日韩在线观看| 欧美影院一区二区| 99亚洲视频| 91麻豆国产在线观看| 国产在线视视频有精品| 亚洲永久免费视频| 国产女人水真多18毛片18精品视频| 欧美视频在线一区| 国产亚洲一区二区三区在线播放| 97精品电影院| 国产一区 二区| 天堂成人国产精品一区| 国产精品乱人伦一区二区| 日韩网站在线看片你懂的| 老司机午夜精品视频在线观看| 黄页网站一区| 成人激情小说乱人伦| 久久超级碰视频| 亚洲国产日韩a在线播放| 中文在线资源观看网站视频免费不卡| 欧美精品久久天天躁| 久久久夜夜夜| 国产日韩1区| 影音先锋中文字幕一区| 91美女在线看| 国产白丝精品91爽爽久久| 久久99精品久久只有精品| 日韩在线一二三区| 亚洲欧美另类小说视频| 中文字幕精品在线不卡| ww久久中文字幕| 日韩女优av电影| 91精品国产黑色紧身裤美女| 欧美天堂亚洲电影院在线播放| 榴莲视频成人在线观看| 在线观看亚洲视频啊啊啊啊| 欧美凹凸一区二区三区视频| 成人久久视频在线观看| 国产一区亚洲一区| 久久国产精品色婷婷| 日韩av在线播放中文字幕| 亚洲一区av在线| 亚洲精品高清视频在线观看| 自拍av一区二区三区| 国产精品麻豆一区二区| 亚洲国产精品99久久久久久久久| 久久综合精品国产一区二区三区 | 日本sm残虐另类| 亚洲成av人片在线观看| 亚洲一二三区视频在线观看| 亚洲精品国产高清久久伦理二区| 中文字幕亚洲一区二区va在线| 亚洲国产高清aⅴ视频| 国产日产精品一区| 国产免费久久精品| 国产精品区一区二区三区| 国产精品全国免费观看高清| 国产精品久久久久桃色tv| 国产精品另类一区| 一区在线观看视频| 亚洲免费看黄网站| 一区二区三区精品久久久| 一区二区免费视频| 亚洲午夜日本在线观看| 天天综合网 天天综合色| 天使萌一区二区三区免费观看| 午夜精品123| 青青草伊人久久| 国产一区二区三区在线观看免费视频 | 亚洲日本激情| 99视频日韩| 噜噜噜91成人网| 欧美色老头old∨ideo| 91精品国产福利| 精品精品国产高清a毛片牛牛| 久久麻豆一区二区| 国产精品午夜久久| 夜夜嗨av一区二区三区中文字幕| 丝袜国产日韩另类美女| 精品午夜久久福利影院| 成人激情综合网站| 欧美日韩综合另类|