我们都知道Spring中的BeanFactory是一个IOC容器 , 负责创建Bean和缓存一些单例的Bean对象 , 以供项目运行过程中使用 。
创建Bean的大概的过程:
- 实例化Bean对象 , 为Bean对象在内存中分配空间 , 各属性赋值为默认值
- 初始化Bean对象 , 为Bean对象填充属性
- 将Bean放入缓存
【Spring三级缓存解决循环依赖】而我们创建一个Bean就可以往Map中存入一个Bean 。这时候我们仅需要一个Map就可以满足创建+缓存的需求 。
但是创建Bean过程中可能会遇到循环依赖问题 , 比如A对象依赖了一个B对象 , 而B对象内部又依赖了一个A , 如下:
public class A {B b;}public class B {A a;}
假设A和B我都定义为单例的对象 , 并且需要在项目启动过程中自动注入 , 如下:@Componentpublic class A {@AutowiredB b;}@Componentpublic class B {@AutowiredA a;}
一级缓存- 实例化A对象 。
- 填充A的属性阶段时需要去填充B对象 , 而此时B对象还没有创建 , 所以这里为了完成A的填充就必须要先去创建B对象;
- 实例化B对象 。
- 执行到B对象的填充属性阶段 , 又会需要去获取A对象 , 而此时Map中没有A , 因为A还没有创建完成 , 导致又需要去创建A对象 。
- 这样 , 就会循环往复 , 一直创建下去 , 只到堆栈溢出 。
二级缓存此时我们引入二级缓存用另外一个Map2 {k:name; v:earlybean} 来存储尚未已经开始创建但是尚未完整创建的对象 。
- 实例化A对象之后 , 将A对象放入Map2中 。
- 在填充A的属性阶段需要去填充B对象 , 而此时B对象还没有创建 , 所以这里为了完成A的填充就必须要先去创建B对象 。
- 创建B对象的过程中 , 实例化B对象之后 , 将B对象放入Map2中 。
- 执行到B对象填充属性阶段 , 又会需要去获取A对象 , 而此时Map中没有A , 因为A还没有创建完成 , 但是我们继续从Map2中拿到尚未创建完毕的A的引用赋值给a字段 。这样B对象其实就已经创建完整了 , 尽管B.a对象是一个还未创建完成的对象 。
- 此时将B放入Map并且从Map2中删除 。
- 这时候B创建完成 , A继续执行b的属性填充可以拿到B对象 , 这样A也完成了创建 。
- 此时将A对象放入Map并从Map2中删除 。
AOP是Spring的重要功能 , 实现方式就是使用代理模式动态增强类的功能 。
动态单例目前有两种技术可以实现 , 一种是JDK自带的基于接口的动态Proxy技术 , 一种是CGlib基于字节码动态生成的Proxy技术 , 这两种技术都是需要原始对象创建完毕 , 之后基于原始对象生成代理对象的 。
那么我们发现 , 在二级缓存的设计下 , 我们需要在放入缓存Map之前将代理对象生成好 。
将流程改为:
- 实例化Bean对象 , 为Bean对象在内存中分配空间 , 各属性赋值为默认值
- 如果有动态单例 , 生成Bean对象的代理Proxy对象
- 初始化Proxy对象 , 为Bean对象填充属性
- 将Proxy放入缓存
推荐阅读
- 红茶对胃酸,云顶红茶图片
- 红茶蜂蜜养胃,蜂蜜红茶胃
- AMD|32核Zen 4加持!AMD新一代EPYC处理器现身:缓存翻番
- 红茶枸杞水,红茶能配枸杞和冰糖吗
- 别再自己瞎写工具类了,SpringBoot 内置工具类应有尽有,建议收藏
- 红茶二级三级区别,甘露和绿茶区别
- 春季喝红茶好吗,消化性溃疡喝黑茶好吗
- 发烧喝红茶,发烧了能喝红茶吗
- 红茶特级和特三级,滇江红茶多少钱
- 安溪三级红茶价格,滇江红茶多少钱