Spring是怎么解决循环依赖的?
首先,Spring 解决循环依赖有两个前提条件:
不全是构造器方式的循环依赖
必须是单例
基于上面的问题,我们知道Bean的生命周期,本质上解决循环依赖的问题就是三级缓存,通过三级缓存提前拿到未初始化完全的对象。
第一级缓存:用来保存实例化、初始化都完成的对象
第二级缓存:用来保存实例化完成,但是未初始化完成的对象
第三级缓存:用来保存一个对象工厂,提供一个匿名内部类,用于创建二级缓存中的对象
![](https://img.haomeiwen.com/i26375382/0c0680101dd85179.png)
假设一个简单的循环依赖场景,A、B互相依赖。
![](https://img.haomeiwen.com/i26375382/d39d5e42c5939c82.png)
A对象的创建过程:
创建对象A,实例化的时候把A对象工厂放入三级缓存
![](https://img.haomeiwen.com/i26375382/692d5043a50765af.png)
2.A注入属性时,发现依赖B,转而去实例化B
3.同样创建对象B,注入属性时发现依赖A,一次从一级到三级缓存查询A,从三级缓存通过对象工厂拿到A,把A放入二级缓存,同时删除三级缓存中的A,此时,B已经实例化并且初始化完成,把B放入一级缓存。
![](https://img.haomeiwen.com/i26375382/58d83492cab202e8.png)
4.接着继续创建A,顺利从一级缓存拿到实例化且初始化完成的B对象,A对象创建也完成,删除二级缓存中的A,同时把A放入一级缓存
5.最后,一级缓存中保存着实例化、初始化都完成的A、B对象
![](https://img.haomeiwen.com/i26375382/6245c5d25e315e93.png)
因此,由于把实例化和初始化的流程分开了,所以如果都是用构造器的话,就没法分离这个操作,所以都是构造器的话就无法解决循环依赖的问题了。
网友评论