![](https://i-blog.csdnimg.cn/blog_migrate/df709fc2d5dd361ec449684cebb4f63a.jpeg)
什么是三级缓存?
第一级缓存:也叫单例池,存放已经经历了完整生命周期的Bean对象,可以直接使用的bean对象。
第二级缓存:存放早期暴露出来的单例Bean对象,实例化以后,就把对象放到这个Map中。(Bean可能只经过实例化,属性还未填充),用于解决循环依赖。
第三级缓存:存放早期暴露的单例Bean的工厂,用于解决循环依赖(AOP的场景)。
注意要点:
BeanFactory是单例池
只有单例的bean会通过三级缓存提前暴露来解决循环依赖的问题。非单例的bean,每次使用都会从容器中创建新对象,所以非单例的bean是没有缓存的,不会将其放到三级缓存中。
解决第二级缓存中AOP生成新对象的问题
Spring就提前AOP,比如在加载b的流程中,如果发送了循环依赖,b依赖了a,就要对a执行AOP,提前获取增强以后的a对象,这样b对象依赖的a对象就是增强以后的a了。
二三级缓存意义
就是为了解决循环依赖,且之所以是二三级缓存而不是二级缓存,主要是可以解决循环依赖对象需要提前被aop代理,以及如果没有循环依赖,早期的bean也不会真正暴露,不用提前执行代理过程,也不用重复执行代理过程。
是否可以没有二级缓存
(1)当只有A依赖B,B依赖A 的时候,是可以没有二级缓存的。
(2)当超过2个对象进行相互依赖时,是有可能用到二级缓存的,某些情况下,从三级缓存中获取都是ObjectFactory对象,而通过它创建的实例对象每次可能都不一样的。为了解决这个问题,spring引入的第二级缓存。只用从第二级缓存中获取该对象即可。
是否需要三级缓存
如果没有代理,是可以不需要三级缓存。
如果创建的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