1. 前序
循环依赖就是N个类循环嵌套引用,Spring处理循环依赖问题有三种情况:
- 构造器循环依赖:这种情况Spring无法处理,直接抛出BeanCurrentlyInCreationException异常
- 单例模式下的setter循环依赖:通过“三级缓存”处理循环依赖。
- 非单例循环依赖:无法处理
2. 构造器循环依赖
bean的创建,首先执行类的构造器,将当前正在创建的bean记录在缓存中,singletonsCurrentlylnCreation.add(beanName)。在创建过程中一直存在在缓存中,如果在创建bean过程中发现自己已经在创建bean池中,将抛出BeanCurrentlyInCreationException异常表示循环依赖,对于创建完毕的bean将从创建bean池中移除。
3. 单例模式下的setter循环依赖

spring容器解决单例模式下setter循环依赖,使用了多级缓存:
1. singletonObjects:缓存生成的单例对象
2. singletonFactories:缓存提前暴露的生成半成品bean的ObjectFactory
3. earlySingletonObjects:存储由ObjectFactory生成的半成品bean
对于A->B,B->A的情况,spring容器在生成A的过程中,首先暴露ObjectFactory到缓存singletonFactories,然后在容器中查找依赖B。容器生成依赖B,使用深度递归调用,调用getBean(name)。在生成B的过程中依赖A,在容器中查找A,首先在缓存中查找。在缓存singletonFactories中获取提前暴露的生成A的ObjectFactory,生成返回A的引用,B对象生成完成返回,继而A对象生成完成。
spring容器使用提前暴露半生成bean、多级缓存机制,实现解决单例模式下setter的循环依赖问题。
4. 非单例循环依赖
对弈prototype作用域的bean,spring容器无法完成依赖注入,因为spring容器不进行缓存prototype作用域的bean,因此无法提前暴露一个创建中的bean