0X01、前提条件
-
bean为单例模式,其他模式暂且不表。
-
Bean工厂支持循环依赖即allowCircularReferences=true,这是它的默认值 。
0X02、解决方法
1、 bean工厂在创建bean之前会尝试从缓存中拿到bean,bean工厂中定义了三级缓存:
singletonObjects:一级缓存
earlySingletonObjects:二级缓存
singletonFactories:三级缓存
以下代码片段摘自DefaultSingletonBeanRegistry类
protected Object getSingleton(String beanName, boolean allowEarlyReference) {
//1、优先尝试从单例缓存中获取bean
Object singletonObject = this.singletonObjects.get(beanName);
//单例缓存中拿不到bean并且这个bean还在创建中
if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
synchronized (this.singletonObjects) {
//2、其次尝试从早期单例缓存中获取bean
singletonObject = this.earlySingletonObjects.get(beanName);
if (singletonObject == null && allowEarlyReference) {
/*3、再次尝试使用对象工厂获取bean,经过对象工厂获取的bean,会经过后置处理器进行处理
为了提升执行效率,避免多次对bean进行处理,从对象工厂中获取的bean会放入到早期单例缓存中。
当前bean还在创建中的时候,遇到循环依赖的情况时,第一次会从对象工厂中获取bean,后面其他的bean如果继续依赖当前bean,会从早期单例缓存中拿到bean
当前bean已经创建完成的时候,遇到依赖bean的情况时,会直接从单例缓存中拿到bean。
*/
ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
if (singletonFactory != null) {
singletonObject = singletonFactory.getObject();
this.earlySingletonObjects.put(beanName, singletonObject);
this.singletonFactories.remove(beanName);
}
}
}
}
return (singletonObject != NULL_OBJECT ? singletonObject : null);
}
2、 首次创建bean的时候,从缓存中是拿不到东西的,所以需要创建,bean在创建之前先对其标记为正在创建:
//bean在创建之前先对它做标记,表示当前bean正在创建中
beforeSingletonCreation(beanName);
3、 正式创建bean对象,创建完成后将bean用对象工厂对象封装下放入到 singletonFactories对象工厂缓存中
//将当前bean的对象工厂放入到bean工厂缓存中。
addSingletonFactory(beanName, new ObjectFactory<Object>() {
@Override
public Object getObject() throws BeansException {
return getEarlyBeanReference(beanName, mbd, bean);
}
});
4、对象创建完成后,需要处理其依赖属性。假设当前bean为A,A依赖B,B又依赖A,对B注入A属性时,可以从三级缓存中拿到bean,拿出之后会将当前bean放入二级缓存中,这样就可以解决循环依赖,记住此时A对象仍然是正在创建标记。还有另一种情况,A依赖B,B依赖A同时又依赖C,C又依赖A,假设先对B注入A属性,再对C注入A属性,此时C是从二级缓存中拿出来的。
5、将当前bean放入一级缓存中,并移除正在创建标记。
0X03 总结
bean正在创建情况下:只有一次机会从三级缓存中拿到bean,从三级缓存中获取的bean是需要经过后置处理处理的,为避免重复处理,需要将其放入到二级缓存中,其他时候都是从二级缓存中拿到的。
bean创建完成后,都是从一级缓存中拿到bean。