Spring为什么要用三级缓存来解决循环依赖

缓存的实质是map

首先为什么会出现循环依赖?见下图

 即在创建A类时需要填充B类的属性,创建B类时需要填充A类的属性

此时A类和B类则产生了循环依赖。

初步解决方案:

         即在创建一个普通对象后,创建一个额外的Map来存储该类的普通对象,以便依赖的另一个类进行调用。但此二级缓存在初始化后的AOP就会出现问题(第五步),实质上依赖的另一个类要获取的是该类的代理对象属性,而不是普通对象属性。所以正确的二级缓存如下图所示:

         即出现循环依赖后,在第一级缓存(单例池)中没有找到代理对象,则创建第二级缓存(earlySingletonObjects),并提前AOP,存放一个临时的代理对象。

         二级缓存(EarlySingletonObjects)的作用:在出现循环依赖后,保证提前AOP后生成的代理对象是唯一的。其实一方面也存着没有完全属性的代理对象,因为此时的代理对象的属性还没有经过bean的完整生命周期。即二级缓存中存放的是没有经过完整的生命周期的bean

 但为什么又需要三级缓存呢?

          流程:第一级缓存单例池中找不到对象,则会进入到creatingSet以此出现循环依赖,然后在第二级缓存earlySingletonObjects中找,若第二级缓存找不到则会在第三级缓存singletonFactories中找,找不到则会在三级缓存中创建一个lambda表达式。这个表达式lambda若有需要处理的AOP逻辑则会提前AOP,生成一个代理对象存入二级缓存,若没有需要处理的AOP,则生成一个普通对象存入二级缓存。详情见下图所示:

​​​​​​​

各级缓存的作用:

一级缓存singletonObjects:单例池。经历一个完整的bean的生命周期之后,bean对象存入单例池。

二级缓存earlySingletonObjects:用来存储没有经历过完整的bean的生命周期的单例bean。保证单例。

三级缓存singletonFactories:存lambda表达式,某一个bean是否会出现循环依赖一开始是不知道的,三级缓存的出现让解决循环依赖的步骤更加方便。若没有三级缓存则不太好进行AOP。即存lambda表达式以进行AOP(普通对象or代理对象)

源码如下图所示:

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值