Spring 是怎么解决循环依赖的?

1. 初始化流程启动:

   Bean A 触发初始化,但在初始化过程中发现依赖了 Bean B。

   Bean B 同时也触发初始化,但在初始化过程中发现依赖了 Bean C。

2. 循环依赖检测:

   在初始化 Bean C 的过程中,发现需要依赖 Bean A。

   Spring容器通过三级缓存的机制来解决循环依赖。三级缓存包括`singletonObjects`(一级缓存)、`earlySingletonObjects`(二级缓存)和`singletonFactories`(三级缓存)。

3. 提前曝光:

   在初始化 Bean A 的过程中,由于使用了`ObjectFactory`提前曝光,将 A 提前放入了`singletonFactories`(三级缓存)中。

4. 循环依赖解决:

   当初始化 Bean C 需要依赖 Bean A 时,通过`ObjectFactory#getObject()`方法从`singletonFactories`中拿到 A 对象,从而避免了循环依赖的问题。

   Bean C 完成初始化后,将自己添加到一级缓存(`singletonObjects`)中。

5. 回到 B:

   Bean B 通过初始化流程,此时可以正常获取 Bean C,完成初始化。

   Bean A 也能正常获取 Bean B,最终整个链路完成初始化过程。

这种通过提前曝光和三级缓存的机制,Spring容器能够有效地解决循环依赖的问题。

  • 8
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
源码深度解析是一种深入研究源代码的方法,通过仔细阅读和理解源代码中的细节和逻辑,以获得对代码的深刻理解和洞察。这样的分析可以帮助开发者更好地理解代码的实现方式,从而更好地理解并使用该代码库。 关于spring如何解决循环依赖的问题,我们可以从源码的角度来分析。Spring采用了三级缓存来解决循环依赖的问题。 第一级缓存是singletonFactories缓存,用于存储正在创建的Bean的工厂对象。当容器正在创建一个Bean时,会将这个Bean的工厂对象存储在singletonFactories缓存中。 第二级缓存是earlySingletonObjects缓存,用于存储已经完成了属性填充但尚未初始化完成的Bean。当容器创建一个Bean时,会将正在创建的Bean存储在earlySingletonObjects缓存中。 第三级缓存是singletonObjects缓存,用于存储已经完成初始化的Bean。当一个Bean初始化完成后,会将其存储在singletonObjects缓存中。 Spring在创建Bean的过程中,会先查找一级缓存,如果找到了对应的工厂对象,则直接返回该对象,避免了创建过程中的循环依赖。如果一级缓存中没有找到对应的工厂对象,则通过递归的方式创建依赖的Bean。 在创建Bean的递归过程中,如果发现正在创建的Bean已经在二级缓存中,说明发生了循环依赖。此时,Spring会从二级缓存中获取正在创建的Bean的代理对象,以解决循环依赖。 当一个Bean创建完成后,会将其放入三级缓存中,并从一级缓存和二级缓存中移除。 总结来说,Spring通过三级缓存的方式解决循环依赖的问题,保证了Bean的创建过程中不会陷入无限递归的循环。这种机制的实现使得Spring解决循环依赖问题上具有较好的性能和效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

郭梓航

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值