一、什么是循环依赖
循环依赖也叫循环引用,是指bean之间形成相互依赖的关系,由此,bean对象在属性注入时便会产生循环。这种循环依赖会导致编译器无法编译代码,从而无法运行程序。为了避免循环依赖,我们在开发过程中需要避免类之间的过度耦合,使用接口和抽象类来降低类之间的依赖性,同时避免在类的构造函数中创建依赖的对象。循环依赖如下图所示:
当然两个以上注入也可能出现循环依赖问题。
我们把这个流程放到Bean的生命周期中去看一下
当然这种依赖不仅仅在两个bean对象中产生,也可能会发生在一个或多个bean对象中。
二、解决方案
Spring通过三级缓存机制解决循环依赖问题。
三级缓存机制
Spring的三级缓存机制指的是在 Spring 容器中创建 bean 对象时,Spring 容器在创建过程中会维护三个缓存区,分别是 Singleton 缓存区、Early Singleton 缓存区和 Factory Bean 缓存区。
一级缓存(singletonObjects):用于保存实例化,注入,初始化完成的bean,一级缓存中的bean对象是完整的可供使用的bean对象
二级缓存(earlySingletonObjects):用于保存实例化完成的bean对象,此时的bean对象,可能被一个或多个bean依赖。
三级缓存(singletonFactories):用于保存实例化完成,但未初始化的bean,此时的bean对象无法使用。
思考题:二级缓存完全够用,为什么是三级缓存?
三、Spring无法解决的循环依赖
- 1.多例Bean通过setter注入的情况,不能解决循环依赖问题
- 2.构造器注入的Bean的情况,不能解决循环依赖问题
- 3.单例的代理Bean通过Setter注入的情况,不能解决循环依赖问题
- 4.设置了@DependsOn的Bean的情况,不能解决循环依赖问题