在Spring框架中,循环依赖(Circular Dependency)是一个常见的概念,它指的是两个或多个bean相互依赖对方,形成了一个闭环。例如,Bean A依赖于Bean B,而Bean B又依赖于Bean A。Spring容器默认可以处理单例(singleton)作用域下的构造器注入(constructor injection)的循环依赖,但是通过setter注入(setter injection)或原型(prototype)作用域下的循环依赖则可能导致问题。
Spring处理单例作用域下的构造器注入循环依赖的方式是通过三级缓存来实现的:
- SingletonObjects:一级缓存,存储完全初始化好的bean,即可以直接使用的bean。
- EarlySingletonObjects:二级缓存,存储bean的早期引用(即实例化后但尚未填充属性的对象),用于解决循环依赖。
- SingletonFactories:三级缓存,存储用于生成bean的ObjectFactory。
当Spring容器遇到循环依赖时,它会这样处理:
- A正在创建中,将A的ObjectFactory放入三级缓存中。
- A请求B,容器检查B是否创建完成:
- 如果B已经创建完成(在一级缓存中),则直接使用。
- 如果B没有创建完成,则检查三级缓存中是否有B的ObjectFactory:
- 如果有,从ObjectFactory中获取B的早期引用(此时B只实例化但尚未填充属性),并将其注入到A中。同时,将B的ObjectFactory从三级缓存移动到二级缓存中。
- 如果没有,则创建B,并重复上述过程。
- 当A填充完属性后,将其放入一级缓存中,此时A的引用是完整的。
需要注意的是,Spring的循环依赖检测和处理主要关注单例作用域下的构造器注入。对于原型作用域或setter注入的循环依赖,Spring通常无法自动解决,并会抛出异常。在设计应用程序时,应尽量避免循环依赖,因为它可能导致代码难以维护和理解。如果确实需要循环依赖,可以考虑使用setter注入或重构代码以消除循环依赖。