一、什么是循环依赖?
A对象依赖B对象,B对象依赖A对象。
//循环依赖
class A{
public B b;
}
class B{
public A a;
}
//循环依赖
A a = new A();
B b = new B();
a.b=b;
b.a=a;
二、Spring中的循环依赖有什么特殊的?
在Spring中,对象并不是简单的New出来的,而是要经过一系列Bean的生命周期,Spring循环依赖特殊就特殊在Bean的生命周期中。
三、Bean的生命周期?(简述)
被Spring管理的对象叫做Bean,在Spring中,Bean是如何生成的?
1、Spring扫描class文件得到BeanDefinition。
2、根据得到的BeanDefinition去生成Bean。
3、根据class推断出构造方法。
4、根据推断的构造方法,反射,得到一个原始对象。
5、向原始对象中填充属性(依赖注入)。
6、如果原始对象中某个方法被AOP了,那么需要根据原始对象生成一个代理对象。
7、把代理对象放入单例池(SingletonObjects),getBean时直接从单例池获取。
更详细的Bean生命周期请看:https://www.cnblogs.com/javazhiyin/p/10905294.html
四、循环依赖特殊点
在依赖注入时,比如上面的A、B类,A类中存在一个B类的b属性,当A类生成了一个原始对象之后,就会去给b属性去赋值,此时就会根据b属性的类型和属性名去BeanFactory中去获取B类所对应的单例bean。如果此时BeanFactory中存在B对应的Bean,那么直接拿来赋值给b属性;如果此时BeanFactory中不存在B对应的Bean,则需要生成一个B对应的Bean,然后赋值给b属性。
那么问题来了,如果BeanFactory不存在B对应的Bean,那么就会去进行B的生命周期,然后B中又存在A类的a属性......
那么应该如何解决这个无限循环问题呢,Spring中为我们提供了一个叫做三级缓存的机制。
五、 三级缓存
一级缓存为:singletonObjects:已经经历了完整生命周期的bean对象。
二级缓存为:earlySingletonObjects:缓存的是早期的bean对象,表示Bean的生命周期还没走完就把这个Bean放入了。
三级缓存为:singletonFactories:缓存的是ObjectFactory,表示对象工厂,用来创建某个对象的。
上面的问题为:
A创建时--->需要B---->B去创建--->需要A,从而产生了循环。
解决方法是:在A创建时,在A进行依赖注入之前,将A的原始对象放入缓存中,放入缓存后在进行依赖注入,创建B的Bean时,在依赖注入之前将B的原始对象放入缓存,这样,当他们之间相互需要时就可以去缓存中拿到。