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无法解决,需要开发者手动优化设计,避免构造函数依赖的循环。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值