Spring 为什么要三级缓存,二级缓存不行吗?

Spring 设计了三级缓存来解决循环依赖问题。

  1. singletonObjects 一级缓存,用于保存实例化、注入、初始化完成的 bean 实例
  2. earlySingletonObjects 二级缓存,用于保存实例化完成、还没依赖注入的 bean 实例
  3. singletonFactories 三级缓存,用于保存 bean 创建工厂,以便于后面扩展有机会创建代理对象。

三级缓存的核心思想,就是把 Bean 的实例化和依赖注入进行分离

举个例子,testService1、testService2 互相依赖,spring 是如何解决循环依赖的:
在这里插入图片描述

图片来自 spring为什么使用三级缓存而不是两级? - 知乎 (zhihu.com)

为什么要三级缓存,二级缓存不行?

其实也行,并不是说二级缓存如果存在aop的话就无法将代理对象注入的问题,本质应该说是初始spring是没有解决循环引用问题的,设计原则是 bean 实例化、属性设置、初始化之后再生成aop对象,但是为了解决循环依赖但又尽量不打破这个设计原则的情况下,使用了存储了函数式接口 ObjectFactory 的第三级缓存。

如果使用二级缓存的话,可以将 AOP 代理工作提前到 提前暴露实例 的阶段执行; 也就是说所有的 bean 在创建过程中就直接生成代理对象;但是这样的话,就和 Spring AOP 的设计原则相驳,AOP 的实现需要与 bean 的正常生命周期的创建分离

比如下面是一个二级缓存,对应三级缓存中的前两级缓存。现在有两个类 A、B 互相依赖,且 B 需要 AOP 代理增强。

步骤:

  1. A 从一级缓存中取不到实例,创建实例 A,添加进二级缓存
  2. A 进行依赖注入,要注入 B
  3. B 从一级缓存中取不到实例,直接创建代理类 B 的实例,添加进二级缓存
  4. 代理 B 进行依赖注入,要注入 A
  5. A 从二级缓存中获取到代理 B,A 依赖注入完成
  6. A 初始化完成,A 添加到一级缓存,删除二级缓存中的 A
  7. 代理 B 从一级缓存中获取到 A,代理 B 依赖注入成功
  8. 代理 B 初始化完成,代理 B 添加到一级缓存,删除二级缓存中的代理 B

在这里插入图片描述

那怎样才能让 AOP 的实现需要与bean的正常生命周期的创建分离?

使用第三级缓存封装一个函数式接口对象到缓存中, 发生循环依赖时,触发代理类的生成。也就是依赖注入前,发现依赖的对象是需要代理增强的,从第三级缓存中获取代理类存入第二级缓存,然后继续完成依赖注入,把代理增强和 bean 生命周期的创建分离开。

还是上面的例子,如果改成三级缓存步骤如下:

  1. A 从一级缓存和二级缓存中取不到实例,创建 A 工厂,添加进三级缓存
  2. A 进行依赖注入,要注入 B
  3. B 从一级缓存和二级缓存中取不到实例,创建 B 工厂,添加进三级缓存
  4. B 进行依赖注入,要注入 A
  5. A 发现要注入的 B 需要代理增强,从三级缓存中获取到 B 工厂,B 工厂创建代理 B 实例,并把 代理 B 存入二级缓存,A 依赖注入完成
  6. A 初始化完成,A 添加到一级缓存,删除二级缓存中的 A 工厂
  7. 代理 B 从一级缓存中获取到 A,代理 B 依赖注入成功
  8. 代理 B 初始化完成,代理 B 添加到一级缓存,删除二级缓存中的代理 B 和 三级缓存中的 B 工厂

当然这都是理论,要验证还需要代码的实践,这位老哥对源码进行了修改,证实了二级缓存下,Spring 的确是可以运行正常。👇
【超级干货】为什么spring一定要弄个三级缓存?

  • 1
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值