当前位置:首页 > 科技  > 软件

Spring到底是如何解决循环依赖问题的?​

来源: 责编: 时间:2023-11-28 09:33:33 342观看
导读作者 | 波哥审校 | 重楼Spring作为当前使用最广泛的框架之一,其重要性不言而喻。所以充分理解Spring的底层实现原理对于咱们Java程序员来说至关重要,那么今天笔者就详细说说Spring框架中一个核心技术点:如何解决循环依赖

作者 | 波哥2Th28资讯网——每日最新资讯28at.com

审校 | 重楼2Th28资讯网——每日最新资讯28at.com

Spring作为当前使用最广泛的框架之一,其重要性不言而喻所以充分理解Spring的底层实现原理对于咱们Java程序员来说至关重要,那么今天笔者就详细说说Spring框架中一个核心技术点:如何解决循环依赖问题?2Th28资讯网——每日最新资讯28at.com

2Th28资讯网——每日最新资讯28at.com

什么是循环依赖问题?

Spring的循环依赖问题是指在使用Spring容器管理Bean的依赖关系时,出现多个Bean之间相互依赖,形成一个循环的依赖关系。这意味着Bean A 依赖于Bean B,同时Bean B 也依赖于Bean A,从而形成一个循环。Spring容器需要确保这些循环依赖关系被正确解决,以避免初始化Bean时出现问题。2Th28资讯网——每日最新资讯28at.com

如果你去网上搜索“Spring是如何解决循环依赖问题的”,绝大部分答案都是:Spring使用三级缓存确保循环依赖的解决,包括"singletonObjects"、"earlySingletonObjects"和"singletonFactories"等缓存,以及占位符的使用等等。这当然没有错,可是看到这些文章的朋友们,你们真的理解了这其中的原理吗?还是只是会背答案呢?那么今天笔者就来扒一扒Spring是如何解决这一问题的底层实现原理。当然要明白这个问题的底层实现原理,你得有一定的Spring源码基础才行哦。2Th28资讯网——每日最新资讯28at.com

现在假设我们有三个类,ClasssA、ClassB、ClassC,代码如下:2Th28资讯网——每日最新资讯28at.com

2Th28资讯网——每日最新资讯28at.com

下面我们根据Spring关于Bean的生命周期管理过程进行分析:2Th28资讯网——每日最新资讯28at.com

假设首先实例化ClassA我们知道在ClassA实例化完成后,需要填充属性classB,在填充classB属性之前,会调用addSingletonFactory方法,把一个Lambda表达式添加到了singletonFactories集合中,这个Lambda表达式的代码如下:2Th28资讯网——每日最新资讯28at.com

2Th28资讯网——每日最新资讯28at.com

2Th28资讯网——每日最新资讯28at.com

2Th28资讯网——每日最新资讯28at.com

在填充属性时,需要获取到classB的实例对象,也就是说会调用getBean("classB")来走classB这个bean实例的生命周期流程。2Th28资讯网——每日最新资讯28at.com

在获取classB实例时首先会调用getSingleton从singletonObjects获取(而这个singletonObjects就是我们平常所说的单例池, 其实就是个map集合):2Th28资讯网——每日最新资讯28at.com

2Th28资讯网——每日最新资讯28at.com

2Th28资讯网——每日最新资讯28at.com

如果单例池中没有才会去创建,那么此时单例池中肯定没有ClassB的实例,所以针对classB实例也会走一遍创建实例的生命周期的流程,同样的也会把上述Lambda表达式添加到singletonFactories集合中。2Th28资讯网——每日最新资讯28at.com

此时singletonFactories集合中就有了classA和classB的两个表达式。2Th28资讯网——每日最新资讯28at.com

但是这里我们要特别注意classB中需要填充属性classA,所以在填充classB实例的classA属性时,同样需要调用getBean("classA")方法来获取到classA的实例,在获取classA实例时,同样首先会调用getSingleton从单例池中获取:2Th28资讯网——每日最新资讯28at.com

2Th28资讯网——每日最新资讯28at.com

如代码所示,首先会根据beanName从singletonObjects获取,也就是获取classA,很显然,classA还没有放到单例池里面去,只有完全创建好的实例才会放到单例池里面去。可以看到代码同时执行
isSingletonCurrentlyInCreation,此时这个方法返回的是true,内容如下:2Th28资讯网——每日最新资讯28at.com

2Th28资讯网——每日最新资讯28at.com

那这个isSingletonCurrentlyInCreation方法是干嘛用的呢?看方法名字就知道了,就是判断当前这个bean是否正在创建中,我们在开始创建classA的时候就已经把他的名字添加到singletonsCurrentlyInCreation这个集合中,表明正在创建classA。2Th28资讯网——每日最新资讯28at.com

很显然满足了if (singletonObject == null &&isSingletonCurrentlyInCreation(beanName))这个条件,于是就进入到if的方法体中。2Th28资讯网——每日最新资讯28at.com

然后从earlySingletonObjects这个集合中获取对象,那这个earlySingletonObjects又是个啥玩意?只用singletonFactories和singletonObjects两个缓存集合不就好了吗?还要多此一举使用earlySingletonObjects干啥呢?是不是感觉没什么用?千万别这么看,大师们考虑问题比咱们要考虑的周到,不服都不行。2Th28资讯网——每日最新资讯28at.com

我们这个案例中ClassA依赖ClassB和ClassC,ClassB依赖ClassA,ClassC也依赖ClassA,假如我们没有这个earlySingletonObjects会出现什么情况呢?我们调用singletonFactories.get(beanName)得到前面说的classA的那个Lambda表达式,然后执行
singletonFactory.getObject()就开始执行这个Lambda表达式,在填充ClassB中的classA属性时是不是相当于执行了这个Lambda表达式获取了这个classA对象。2Th28资讯网——每日最新资讯28at.com

好了,到此为止classA中的classB属性获取到了,接下来填充classC了,上述同样的流程,当填充classC的classA属性时,是不是还得从singletonFactories中获取classA的Lambda表达式,然后再执行那个Lambda表达式,于是执行了两次,正常情况下是没有问题的,因为两个Lambda表达式返回的结果都是classA的实例对象,但是有一种情况下就会有问题了?老铁们此时心中肯定充满疑惑,神马情况呢?2Th28资讯网——每日最新资讯28at.com

如果执行这个Lambda表达式返回的是classA的代理对象呢?如果执行了两次,是不是就表明classB中的classA属性和classC中的classA属性是两个不同的对象了?这问题可就大了,那么问题又来了,神马情况下会返回classA的代理对象?不卖关子了,直接上答案:在classA需要AOP的情况下,是需要生成代理对象的,而这个生成AOP的骚操作就是在这个Lambda表达式中实现的,我们下面会详细介绍。2Th28资讯网——每日最新资讯28at.com

所以这里Spring使用了earlySingletonObjects这个我们称为二级缓存的集合来暂存下,这样在classC填充classA属性的时候就不用再次调用lambda表达式了,是不是完美的解决了上述的问题?剩下的几行代码很简单,就不多废话了,大家自己看看就知道了。2Th28资讯网——每日最新资讯28at.com

总结下,Spring解决循环依赖问题其实就是使用了几个集合类,它们分别是:singletonsCurrentlyInCreation(SetsingletonFactories(Map)earlySingletonObjects(Map)singletonObjects(Map),通过这几个集合的相互配合,最终解决循环依赖问题。2Th28资讯网——每日最新资讯28at.com

作者介绍

波哥,互联行业从业10余年,先后担任项目总监及架构师。目前专攻技术,喜欢研究技术原理。技术全面,主攻Java,精通JVM底层机制及Spring全家桶底层框架原理,熟练掌握当前主流的中间件、服务网格等技术原理。2Th28资讯网——每日最新资讯28at.com


2Th28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-34591-0.htmlSpring到底是如何解决循环依赖问题的?​

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 使用 sync.Cond 来协调并发 goroutine 的访问共享资源

下一篇: 聊聊 Calico BGP容器网络实践

标签:
  • 热门焦点
  • 对标苹果的灵动岛 华为带来实况窗功能

    继苹果的灵动岛之后,华为也在今天正式推出了“实况窗”功能。据今天鸿蒙OS 4.0的现场演示显示,华为的实况窗可以更高效的展现出实时通知,比如锁屏上就能看到外卖、打车、银行
  • 帅气纯真少年!日本最帅初中生选美冠军出炉

    日本第一帅哥初一生选美大赛冠军现已正式出炉,冠军是来自千叶县的宗田悠良。日本一直热衷于各种选美大赛,从“最美JK”起到“最美女星&r
  • 2023 年的 Node.js 生态系统

    随着技术的不断演进和创新,Node.js 在 2023 年达到了一个新的高度。Node.js 拥有一个庞大的生态系统,可以帮助开发人员更快地实现复杂的应用。本文就来看看 Node.js 最新的生
  • 分享六款相见恨晚的PPT模版网站, 祝你做出精美的PPT!

    1、OfficePLUSOfficePLUS网站旨在为全球Office用户提供丰富的高品质原创PPT模板、实用文档、数据图表及个性化定制服务。优点:OfficePLUS是微软官方网站,囊括PPT模板、Word模
  • 微信语音大揭秘:为什么禁止转发?

    大家好,我是你们的小米。今天,我要和大家聊一个有趣的话题:为什么微信语音不可以转发?这是一个我们经常在日常使用中遇到的问题,也是一个让很多人好奇的问题。让我们一起来揭开这
  • .NET 程序的 GDI 句柄泄露的再反思

    一、背景1. 讲故事上个月我写过一篇 如何洞察 C# 程序的 GDI 句柄泄露 文章,当时用的是 GDIView + WinDbg 把问题搞定,前者用来定位泄露资源,后者用来定位泄露代码,后面有朋友反
  • ESG的面子与里子

    来源 | 光子星球撰文 | 吴坤谚编辑 | 吴先之三伏大幕拉起,各地高温预警不绝,但处于厄尔尼诺大“烤”之下的除了众生,还有各大企业发布的ESG报告。ESG是“环境保
  • 华为发布HarmonyOS 4:更好玩、更流畅、更安全

    在8月4日的华为开发者大会2023(HDC.Together)大会上,HarmonyOS 4正式发布。自2019年发布以来,HarmonyOS一直以用户为中心,经历四年多的发展HarmonyOS已
  • 7月4日见!iQOO 11S官宣:“鸡血版”骁龙8 Gen2+200W快充加持

    上半年已接近尾声,截至目前各大品牌旗下的顶级旗舰都已悉数亮相,而下半年即将推出的顶级旗舰已经成为了数码圈爆料的主流,其中就包括全新的iQOO 11S系
Top