Spring源码一.(7)循环依赖

1.背景

普通循环依赖的问题点

存在栈溢出现象

2.spring中的支持循环嘛?

支持属性setter注入方式的循环依赖,通过三级缓存来实现支持的

2.1.一级缓存

private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);

2.2.二级缓存

private final Map<String, Object> earlySingletonObjects = new ConcurrentHashMap<>(16);

2.3.三级缓存

private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);

 

3.spring是如何解决?

自己理解的流程文本!仅仅记录

第一次调用getBean方法:getBean(a)

第一次:Object sharedInstance = getSingleton(beanName);

实例化A时,sharedInstance为空,因为正在创建的set集合中没它

markBeanAsCreated(beanName); 把A标记为正在创建,丢A的名字到set集合:alreadyCreated中

利用bean后置处理器的第四次调用代码处:存入spring的三级缓存中。(this.singletonFactories.put(beanName, objectFactory);)

然后A的不完整实例对象,进行属性填充,populateBean

B...对象按照上述流程过一次,B对象进行属性填充A,

第二次进入到getBean方法这里:getBean(a)

第二次:Object sharedInstance = getSingleton(a); 正在创建的set集合中拥有它了:isSingletonCurrentlyInCreation(a)

one:首先从一级缓存中拿,拿不到

two:其次从二级缓存中拿,也拿不到

three:然后从三级缓存中拿,拿到一个objectFactory.

four: 调用对象工厂的 singletonFactory.getObject()拿到一个 尚未属性填充的A实例对象,然后移除三级缓存

接着又会到B对象,属性填充完a,进行b的初始化,接着往一级缓存中放入B,并且二,三级缓存中移除B

接着A的属性注入b继续走:把B对象的实例注入到A的成员属性中

然后A对象进行初始化,一样往一级缓存中放入A,并且二,三级缓存中移除A

4.部分DEMO

 

4.总结

其实这里就是利用了一个 缓存的思想来解决循环依赖的问题,只不过spring中的循环依赖要考虑的东西比较多,这个缓存就被设计成了三级的形式!

一级缓存:存储完整的bean对象

二级缓存:存储的是尚未属性注入的纯净bean,可防止多线程情况下获取到不完整bean的情况

三级缓存:存储了一个函数接口,只要调用objectFactory.getObject方法就返回一个对象

注意:当循环依赖的情况下,要实现A对象的动态代理的话?spring是如何实现的?

正常情况下A对象的动态代理是在spring中bean生命周期第八次调用BeanPostprocessor对象的初始化后方法执行的!

但是当存在循环依赖的情况下,A对象要被提前注入都B对象中。此时如果还是注入普通的A对象,那么不合理! 所以A对象的AOP代理被提前到了实例化后阶段。这里因为三级缓存存储的是一个函数接口,当被调用时,会去判断A对象是否需要代理,如果需要就去走AOP进行提前代理,然后注入到B的成员属性之中

 

5.常问的热点问题

5.1.spring是如何解决循环依赖的?

一,二,三级缓存

5.2. 为啥需要三级缓存?

嘻嘻

5.3. 构造方法形式的循环依赖支持不? 多例情况下的循环依赖支持不?

不支持,不支持

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值