spring解决循环依赖

spring是利用三级缓存来解决循环依赖问题的

首先要明白三级缓存其实是三个Map

  • 一级缓存:SingletonObjects,一个ConcurrentHashMap<String,Object>(也叫作单例池)
  • 二级缓存:EarlySingletonObjects,一个ConcurrentHashMap<String,Object>
  • 三级缓存:SingletonFactories,一个HashMap<String,ObjectFactory<?>>

什么是循环依赖呢?
就用对象A和对象B来举例,A中有属性B,B中有属性A

  1. spring在创建bean的时候首先会先创建一个A实例化对象a(其中属性为默认值),
  2. 然后在三级缓存中,创建一个一个key为a(beanName,并不是a对象)、value为一个lambda表达式(()->getEarlyBeanReference(…)),
  3. 这时候去给a的属性b赋值,先从一级缓存中看有没有B的对象,没有的话再去二级缓存中取,再去发现三级缓存中看,都没有那么就会也去创建一个B的实例化对象b(其中属性为默认值),
  4. 然后一样在然后在三级缓存中,创建一个一个key为b(beanName,并不是a对象)、value为一个lambda表达式(()->getEarlyBeanReference(…)),
  5. 这时候去给b的属性a去赋值,先从一级缓存中看有没有A的对象,没有的话再去二级缓存中取,再去发现三级缓存中有A的对象证明A正在创建中,那么就会从三级缓存中取出key为a的lambda表达式调用getEarlyBeanReference方法,生成了具体的a对象(如果有AOP,那在这将代理对象把原来对象覆盖掉)放到二级缓存中(这时a是个半成品对象,a中的b还是没有值的)。(为什么需要三级缓存?来保证同名对象只有一个,bean的创建流程中肯对会提前生成一个普通对象,因为不知道对象什么时候引用,也就不知道对象什么时候代理好,所以只能在第一次引用对象的时候来判断对象是否需要被代理,所以通过lambda表达式的回调,来判断是否要将普通对象覆盖)
  6. 那么这时b就是一个成品对象了,将对象b放入一级缓存中,将三级缓存中的b删掉
  7. 然后会给a对象中的b属性赋值,这时候a对象也是一个成品对象了,将对象a放入一级缓存中也就是单例池中,将二级缓存中的a对象删掉

总结
如果说当代理对象存在循环依赖问题那么需要三级缓存,如果没有代理对象的产生的循环依赖,那么只需要二级缓存就可以了

个人理解,可能理解的不是很到位,希望一起探讨

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值