Spring是如何解决循环依赖?

现象解释:
在Spring框架中,循环依赖(Circular Dependency)是指两个或多个Bean之间相互依赖,形成了一个循环。例如,Bean A依赖于Bean B,而Bean B又依赖于Bean A。Spring通过多种机制解决循环依赖问题,具体来说,主要有以下几种方式:

1.三级缓存机制

Spring容器在实例化Bean时使用了三级缓存来解决循环依赖,主要涉及三个缓存结构:

  • 一级缓存:singletonObjects,保存完全初始化好的单例Bean。
  • 二级缓存:earlySingletonObjects,保存提前曝光的单例Bean,用于防止循环依赖。
  • 三级缓存:singletonFactories,保存能够生成Bean实例的工厂,用于创建Bean的早期引用。

解决循环依赖的流程:

  1. 创建实例:Spring首先通过反射调用Bean的构造方法,创建一个半成品Bean实例,但并不立即进行属性填充。
  2. 三级缓存注册:将这个半成品Bean的创建工厂(ObjectFactory放入三级缓存singletonFactories中,以便在必要时通过工厂获取Bean实例。
  3. 属性注入:当注入依赖的Bean时,Spring会先查看依赖Bean是否已经存在于一级缓存中。如果没有,检查二级缓存和三级缓存。如果Bean存在于三级缓存中,则通过ObjectFactory生成早期Bean引用,注入到其他Bean中,避免循环依赖。
  4. 完全初始化:当依赖注入完成后,Bean会被完全初始化,并且从三级缓存逐渐移动到二级缓存、一级缓存,最终完成Bean的创建。

2.代理Bean的提前曝光

在某些场景下,Spring可能使用动态代理提前暴露Bean。这种方式允许通过代理引用Bean,而不是真实对象本身,这样即使对象尚未完全创建,也可以被注入其他Bean中,解决了循环依赖的问题。

3.依赖查找注入(Autowired的setter方式)

对于通过@Autowired注解的setter方法注入Bean,Spring在创建Bean时可以推迟某些属性的注入,避免了在Bean构造时立即注入依赖,进而解决循环依赖。

4.构造函数依赖无法解决的循环依赖

Spring只能解决基于setter注入的循环依赖。如果是构造函数注入的循环依赖(即A的构造函数依赖B,B的构造函数又依赖A),则无法通过上述方式解决,因为在构造函数执行时必须提供完整的依赖,Spring会抛出BeanCurrentlyInCreationException。

5.小结

  • setter注入循环依赖:Spring通过三级缓存机制、早期暴露Bean引用来解决。
  • 构造函数注入循环依赖:Spring无法解决,需要开发者手动优化设计,避免构造函数依赖的循环。
源码深度解析是一种深入研究源代码的方法,通过仔细阅读和理解源代码中的细节和逻辑,以获得对代码的深刻理解和洞察。这样的分析可以帮助开发者更好地理解代码的实现方式,从而更好地理解并使用该代码库。 关于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
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值