循环依赖
循环依赖其实就是循环引⽤,也就是两个或者两个以上的 Bean 互相持有对⽅,最终形成闭环。⽐如A
依赖于B,B依赖于C,C⼜依赖于A。
Spring 中循环依赖场景有:
- 构造器的循环依赖(构造器注⼊)
- Field 属性的循环依赖(set注⼊)
其中,构造器的循环依赖问题⽆法解决,只能拋出BeanCurrentlyInCreationException
异常,在解决
属性循环依赖时,Spring 采⽤的是提前暴露对象的⽅法。
循环依赖处理机制
prototype
原型bean
循环依赖(⽆法解决)Spring
对于原型bean
的初始化过程中不论是通过构造器参数循环依赖还是通过setter
⽅法产⽣循环依
赖,Spring
都会直接报错处理。org.springframework.beans.factory.support.AbstractBeanFactory#doGetBean
protected <T> T doGetBean(final String name, @Nullable final Class<T> requiredType,
@Nullable final Object[] args, boolean typeCheckOnly) throws BeansException {
// ...
// Fail if we're already creating this bean instance:
// We're assumably within a circular reference.
if (isPrototypeCurrentlyInCreation(beanName)) {
throw new BeanCurrentlyInCreationException(beanName);
}
// ...
else if (mbd.isPrototype()) {
// It's a prototype -> create a new instance.
Object prototypeInstance = null;
try {
// 原型bean在创建前先标记为正在创建
beforePrototypeCreation(beanName);
prototypeInstance = createBean(beanName, mbd, args);
}
finally {
// 创建完成后将标记清除
afterPrototypeCreation(beanName);
}
bean = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
}
}
protected boolean isPrototypeCurrentlyInCreation(String beanName) {
Object curVal = this.prototypesCurrentlyInCreation.get();
return (curVal != null &&
(curVal.equals(beanName) || (curVal instanceof Set && ((Set<?>) curVal).contains(beanName))));
}
- 单例
bean
构造器参数循环依赖(无法解决)Spring
的循环依赖的理论依据基于Java
的引⽤传递,当获得对象的引⽤时,对象的属性是可以延
后设置的。但是构造器必须是在获取引⽤之前。 - 单例
bean
通过setter
或者@Autowired
进⾏循环依赖
<!--循环依赖问题-->
<bean id="lagouBean" class="com.kirito.LagouBean">
<property name="ItBean" ref="itBean"/>
</bean>
<bean id="itBean" class="com.kirito.ItBean">
<property name="LagouBean" ref="lagouBean"/>
</bean>
解决setter
或者@Autowired
循环依赖其实是通过提前暴露⼀个ObjectFactory
对象。简单来说 A 在调⽤构造器完成对象初始化之后,在调⽤ A 的 setB ⽅法之前就把 A 实例化的对象通过ObjectFactory
提前暴露到Spring
容器中。
具体的实现是通过DefaultSingletonBeanRegistry
持有的三级缓存:
/** Cache of singleton objects: bean name to bean instance. */
// 一级缓存单例池,包含完整的spring bean
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);
/** Cache of singleton factories: bean name to ObjectFactory. */
// 三级缓存,当对象被实例化后立马放入三级缓存
private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);
/** Cache of early singleton objects: bean name to bean instance. */
// 二级缓存早期对象池,当对象从三级缓存中获取到引用对象并赋值后,会升级到二级缓存
private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);
Spring
解决循环依赖过程:
- A 在创建后,立马将自己放入三级缓存池,提前暴露自己;
- A 创建完成后进行属性注入,发现引用了 B,会去缓存中找,没找到就开始创建 B;
- B 在创建后同样会将自己放入三级缓存;
- B 在进行属性注入时发现引用了 A,就去缓存中找,在三级缓存中找到了 A;
- B 升级到二级缓存;(此过程还没有完成依赖注入,可以额外扩展)
- B 在完成属性变成完整的
Spring Bean
后,将自己放入一级缓存; - A 在一级缓存中发现了已经成型的 B,利用 B 继续走创建流程;
- A 创建完成后,也将自己放入一级缓存。
示意图如下:
循环依赖调用流程