1、问题描述:
服务A依赖服务B,服务B又依赖服务A,就形成了循环依赖。再比如A依赖B,B依赖C,C依赖A。
2、spring是如何解决循环依赖的?
spring只能解决单例模式下的循环依赖。解决循环以来的理论依据是基于java的引用传递以及spring框架内定义的三级缓存来实现的,可以说是三级缓存解决了bean之间的循环以来。当我们获取到对象的引用时,对象的属性是可以延后设置的(但是构造器必须在获取引用前)。
spring的单例对象的初始化主要分为三步:
(1)createBeanInstance:实例化,即调用对象的构造方法实例化对象。
(2)populateBean: 填充属性,这一步主要是多bean的依赖属性进行填充。
(3)initializeBean: 初始化,调用spring xml中的init方法。
三级缓存分别是:
(1)singletonObjects: 第一级缓存,里面放置的是实例化好的单例对象。
(2)earlySingletonObjects: 第二级缓存,里面存放的是提前曝光的单例对象。
(3)singletonFactories: 第三级缓存,里面存放的是要被实例化的对象的对象工厂。
所以当一个bean调用构造函数进行实例化后,即使属性还未填充,就可以通过三级缓存向外暴露依赖的引用值(所以循环依赖问题的解决也是基于java的引用传递),这也说明了另外一点,基于构造函数的注入,如果有循环依赖,spring是不能解决的(因为构造函数在引用前调用)。还有,spring默认的bean scope是单例的,而三级缓存中都包含singleton,可见是对于单例bean之间的循环依赖的解决,spring是通过三级缓存来实现的。