电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

spring 如何解決循環依賴

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

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

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

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

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

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

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

這個方法的實現長得很有意思,有著一堆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中獲取,這里面存放的是已經完全創建好的單例Bean;如果取不到,那么就往下走,去earlySingletonObjects里面取,這個是早期曝光的對象;如果還是沒有,那么再去第三級緩存singletonFactories里面獲取,它是提前暴露的對象工廠,這里會從三級緩存里取出后放到二級緩存中。那么總的來說,Spring去獲取一個bean的時候,其實并不是直接就從容器里面取,而是先從緩存里找,而且緩存一共有三級。那么從這個方法返回的并不一定是我們需要的bean,后面會調用getObjectForBeanInstance()方法去得到實例化后的bean,這里就不多說了。

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

// 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這個變量它會是什么值?

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

接下來就進入了if語句的實現里面了,也就是addSingletonFactory()這個方法。看到里面的代碼中出現singletonFactories這個變量是不是很熟悉?翻到上面的getSingleton()就知道了,其實就是三級緩存,所以這個方法的作用是通過三級緩存提前暴露一個工廠對象。

/** * 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); } }}

接下來,回憶下上一章節說的實例化之后的步驟,就是屬性注入了。這就意味著ServiceA需要將ServiceB注入進去,那么顯然又要調用getBean()方法去獲取ServiceB。ServiceB還沒有創建,則也會進入這個createBean()方法,同樣也會來到這一步依賴注入。ServiceB中依賴了ServiceA,則會調用getBean()去獲取ServiceA。此時的獲取ServiceA可就不是再創建Bean了,而是從緩存中獲取。這個緩存就是上面getSingleton()這個方法里面我們看到的singletonFactory。那么這個singletonFactory哪里來的,就是這個addSingletonFactory()方法的第二個參數,即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)的實現,發現有兩個,一個是spring-beans下的SmartInstantiationAwareBeanPostProcessor,一個是spring-aop下的AbstractAutoProxyCreator。我們在未使用AOP的情況下,取的還是第一種實現。

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

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

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

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

好了我們現在是處于創建B的過程,但由于B依賴A,所以調用了獲取A的方法,則A從三級緩存進入了二級緩存,得到了A的代理對象。當然我們不需要擔心注入B的是A的代理對象會帶來什么問題,因為生成代理類的內部都是持有一個目標類的引用,當調用代理對象的方法的時候,實際上是會調用目標對象的方法的,所以所以代理對象是沒影響的。當然這里也反應了我們實際上從容器中要獲取的對象實際上是代理對象而不是其本身。

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

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

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

singletonObject = this.earlySingletonObjects.get(beanName);

因為在前面我們在創建往B中注入A的時候已經從三級緩存取出來放到二級緩存中了,所以這里A可以通過二級緩存去取。再往下就是生命周期后面的代碼了,就不再繼續了。

那么現在就會有個疑問,我們為什么非要三級緩存,直接用二級緩存似乎就足夠了?

看看上面getEarlyBeanReference()這個方法所在的類,它是SpringAOP自動代理的關鍵類,它實現了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緩存中找不到當前的bean的話就會去創建代理。也就是說SpringAOP希望在Bean初始化后進行創建代理。如果我們只使用二級緩存,也就是在這個地方

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

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

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

spring 如何解決循環依賴

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

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

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

依賴注入的方式有setter注入,構造器注入和Field方式。

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

如果A和B都是通過setter注入,顯然對于執行構造器沒有影響,所以不影響解決循環依賴;

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

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

如果A中注入B的方式為構造器,B中注入A為setter,那么這個時候A先進入實例化方法,發現需要B,那么就會去創建B,而A還沒放入三級緩存里,B再創建的時候去獲取A就會獲取失敗。

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

以上就是spring 如何解決循環依賴的詳細內容,更多關于spring 循環依賴的資料請關注好吧啦網其它相關文章!

標簽: Spring
相關文章:
主站蜘蛛池模板: 隧道风机_DWEX边墙风机_SDS射流风机-绍兴市上虞科瑞风机有限公司 | 细胞染色-流式双标-试剂盒免费代做-上海研谨生物科技有限公司 | 济南货架定做_仓储货架生产厂_重型货架厂_仓库货架批发_济南启力仓储设备有限公司 | 喷播机厂家_二手喷播机租赁_水泥浆洒布机-河南青山绿水机电设备有限公司 | 植筋胶-粘钢胶-碳纤维布-碳纤维板-环氧砂浆-加固材料生产厂家-上海巧力建筑科技有限公司 | 干粉砂浆设备-干粉砂浆生产线-干混-石膏-保温砂浆设备生产线-腻子粉设备厂家-国恒机械 | 立式矫直机_卧式矫直机-无锡金矫机械制造有限公司 | hdpe土工膜-防渗膜-复合土工膜-长丝土工布价格-厂家直销「恒阳新材料」-山东恒阳新材料有限公司 ETFE膜结构_PTFE膜结构_空间钢结构_膜结构_张拉膜_浙江萬豪空间结构集团有限公司 | 除尘器布袋骨架,除尘器滤袋,除尘器骨架,电磁脉冲阀膜片,卸灰阀,螺旋输送机-泊头市天润环保机械设备有限公司 | MTK核心板|MTK开发板|MTK模块|4G核心板|4G模块|5G核心板|5G模块|安卓核心板|安卓模块|高通核心板-深圳市新移科技有限公司 | 耐热钢-耐磨钢-山东聚金合金钢铸造有限公司 | 机械立体车库租赁_立体停车设备出租_智能停车场厂家_春华起重 | 精密五金冲压件_深圳五金冲压厂_钣金加工厂_五金模具加工-诚瑞丰科技股份有限公司 | 不锈钢轴流风机,不锈钢电机-许昌光维防爆电机有限公司(原许昌光维特种电机技术有限公司) | 压力变送器-上海武锐自动化设备有限公司 | 四合院设计_四合院装修_四合院会所设计-四合院古建设计与建造中心1 | 成都茶楼装修公司 - 会所设计/KTV装修 - 成都朗煜装饰公司 | 爆破器材运输车|烟花爆竹运输车|1-9类危险品厢式运输车|湖北江南专用特种汽车有限公司 | LED灯杆屏_LED广告机_户外LED广告机_智慧灯杆_智慧路灯-太龙智显科技(深圳)有限公司 | 仓储货架_南京货架_钢制托盘_仓储笼_隔离网_环球零件盒_诺力液压车_货架-南京一品仓储设备制造公司 | 涡轮流量计_LWGY智能气体液体电池供电计量表-金湖凯铭仪表有限公司 | 旋转/数显粘度计-运动粘度测定仪-上海平轩科学仪器 | 耐压仪-高压耐压仪|徐吉电气 | 儿童乐园|游乐场|淘气堡招商加盟|室内儿童游乐园配套设备|生产厂家|开心哈乐儿童乐园 | 天津暖气片厂家_钢制散热器_天津铜铝复合暖气片_维尼罗散热器 | 高光谱相机-近红外高光谱相机厂家-高光谱成像仪-SINESPEC 赛斯拜克 | 山东石英砂过滤器,除氟过滤器「价格低」-淄博胜达水处理 | 缓蚀除垢剂_循环水阻垢剂_反渗透锅炉阻垢剂_有机硫化物-郑州威大水处理材料有限公司 | 正压密封性测试仪-静态发色仪-导丝头柔软性测试仪-济南恒品机电技术有限公司 | 不锈钢轴流风机,不锈钢电机-许昌光维防爆电机有限公司(原许昌光维特种电机技术有限公司) | 黑龙江京科脑康医院-哈尔滨精神病医院哪家好_哈尔滨精神科医院排名_黑龙江精神心理病专科医院 | 江苏皓越真空设备有限公司 | 硬度计,金相磨抛机_厂家-莱州华煜众信试验仪器有限公司 | 宿松新闻网 宿松网|宿松在线|宿松门户|安徽宿松(直管县)|宿松新闻综合网站|宿松官方新闻发布 | 银川美容培训-美睫美甲培训-彩妆纹绣培训-新娘化妆-学化妆-宁夏倍莱妮职业技能培训学校有限公司 临时厕所租赁_玻璃钢厕所租赁_蹲式|坐式厕所出租-北京慧海通 | 合肥白癜风医院_[治疗白癜风]哪家好_合肥北大白癜风医院 | 塑料瓶罐_食品塑料瓶_保健品塑料瓶_调味品塑料瓶–东莞市富慷塑料制品有限公司 | 济南货架定做_仓储货架生产厂_重型货架厂_仓库货架批发_济南启力仓储设备有限公司 | 济南品牌包装设计公司_济南VI标志设计公司_山东锐尚文化传播 | 联系我们老街华纳娱乐公司官网19989979996(客服) | 石磨面粉机|石磨面粉机械|石磨面粉机组|石磨面粉成套设备-河南成立粮油机械有限公司 |