Spring 三级缓存

什么是三级缓存?

第一级缓存:也叫单例池,存放已经经历了完整生命周期的Bean对象,可以直接使用的bean对象。

第二级缓存:存放早期暴露出来的单例Bean对象,实例化以后,就把对象放到这个Map中。(Bean可能只经过实例化,属性还未填充),用于解决循环依赖。

第三级缓存:存放早期暴露的单例Bean的工厂,用于解决循环依赖(AOP的场景)。

注意要点:

  1. BeanFactory是单例池

只有单例的bean会通过三级缓存提前暴露来解决循环依赖的问题。非单例的bean,每次使用都会从容器中创建新对象,所以非单例的bean是没有缓存的,不会将其放到三级缓存中。

  1. 解决第二级缓存中AOP生成新对象的问题

Spring就提前AOP,比如在加载b的流程中,如果发送了循环依赖,b依赖了a,就要对a执行AOP,提前获取增强以后的a对象,这样b对象依赖的a对象就是增强以后的a了。

  1. 二三级缓存意义

就是为了解决循环依赖,且之所以是二三级缓存而不是二级缓存,主要是可以解决循环依赖对象需要提前被aop代理,以及如果没有循环依赖,早期的bean也不会真正暴露,不用提前执行代理过程,也不用重复执行代理过程。

  1. 是否可以没有二级缓存

(1)当只有A依赖B,B依赖A 的时候,是可以没有二级缓存的。

(2)当超过2个对象进行相互依赖时,是有可能用到二级缓存的,某些情况下,从三级缓存中获取都是ObjectFactory对象,而通过它创建的实例对象每次可能都不一样的。为了解决这个问题,spring引入的第二级缓存。只用从第二级缓存中获取该对象即可。

  1. 是否需要三级缓存

如果没有代理,是可以不需要三级缓存。

如果创建的Bean有对应的代理,那其他对象注入时,注入的应该是对应的代理对象;但是Spring无法提前知道这个对象是不是有循环依赖的情况,而正常情况下(没有循环依赖情况),Spring都是在创建好完成品Bean之后才创建对应的代理。这时候Spring有两个选择:

不管有没有循环依赖,都提前创建好代理对象,并将代理对象放入缓存,出现循环依赖时,其他对象直接就可以取到代理对象并注入。

不提前创建好代理对象,在出现循环依赖被其他对象注入时,才实时生成代理对象。这样在没有循环依赖`的情况下,Bean就可以按着Spring设计原则的步骤来创建。

Spring选择了第二种方式,那怎么做到提前曝光对象而又不生成代理呢?

Spring就是在对象外面包一层ObjectFactory,提前曝光的是ObjectFactory对象,在被注入时才在ObjectFactory.getObject方式内实时生成代理对象,并将生成好的代理对象放入到第二级缓存Map<String, Object> earlySingletonObjects。

addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));

对象在三级缓存中的创建流程

场景:A依赖B,B依赖A

1、A创建过程中需要B,于是先将A放到三级缓存,去实例化B。

2、B实例化的过程中发现需要A,于是B先查一级缓存寻找A,如果没有,再查二级缓存,如果还没有,再查三级缓存,找到了A,然后把三级缓存里面的这个A放到二级缓存里面,并删除三级缓存里面的A。

3、B顺利初始化完毕,将自己放到一级缓存里面(此时B里面的A依然是创建中的状态)。然后回来接着创建A,此时B已经创建结束,可以直接从一级缓存里面拿到B,去完成A的创建,并将A放到一级缓存。

更多消息资讯,请访问昂焱数据https://www.ayshuju.com/home

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值