新手做网站流程,正版海外自媒体服务器官网,没有icp许可证 举报一个准吗,网站百度排名怎么做快在Spring框架中#xff0c;循环依赖#xff08;Circular Dependency#xff09;是指多个Bean相互依赖#xff0c;形成一个循环引用。例如#xff0c;Bean A依赖于Bean B#xff0c;而Bean B又依赖于Bean A。这种情况在Bean创建时可能导致Spring容器无法正常完成初始化循环依赖Circular Dependency是指多个Bean相互依赖形成一个循环引用。例如Bean A依赖于Bean B而Bean B又依赖于Bean A。这种情况在Bean创建时可能导致Spring容器无法正常完成初始化抛出错误如下
public class A {private final B b;public A(B b) {this.b b;}
}public class B {private final A a;public B(A a) {this.a a;}
}启动时会出现如下错误
org.springframework.beans.factory.BeanCurrentlyInCreationException:
Error creating bean with name A: Requested bean is currently in creation: Is there an unresolvable circular reference?一、解决循环依赖的方法
1. 构造器注入
构造器注入不支持循环依赖因为Spring在创建Bean时需要解析所有构造函数参数这导致了依赖循环。可以通过使用Lazy注解延迟Bean的初始化来解决此问题Lazy会告诉Spring在第一次使用Bean时才初始化而不是立即初始化。
Component
public class A {private final B b;Autowiredpublic A(Lazy B b) {this.b b;}
}Component
public class B {private final A a;Autowiredpublic B(Lazy A a) {this.a a;}
}2. Setter注入
Setter注入可以解决循环依赖因为Spring可以先创建Bean的实例再注入其依赖。
public class A {private B b;public void setB(B b) {this.b b;}
}public class B {private A a;public void setA(A a) {this.a a;}
}3. 使用Autowired注解
可以使用Autowired进行Setter注入或字段注入同样可以解决循环依赖问题。
public class A {Autowiredprivate B b;
}public class B {Autowiredprivate A a;
}4. 使用Lazy注解
Lazy注解可以延迟Bean的初始化避免循环依赖。
public class A {AutowiredLazyprivate B b;
}public class B {AutowiredLazyprivate A a;
}5. 使用ObjectFactory或Provider
使用ObjectFactory或Provider可以在需要时才获取Bean实例从而解决循环依赖。
public class A {Autowiredprivate ObjectFactoryB bFactory;public void someMethod() {B b bFactory.getObject();// 使用B}
}public class B {Autowiredprivate ObjectFactoryA aFactory;public void someMethod() {A a aFactory.getObject();// 使用A}
}6. 配置allow-circular-references: true
通过allow-circular-references: true配置来允许Spring容器处理Bean之间的循环依赖问题但从设计角度来看尽量避免循环依赖更为合理。
spring:main:allow-circular-references: true二、Spring三级缓存解决循环依赖的原理
Spring在创建Bean时使用三级缓存来处理循环依赖问题。整个过程分为三个阶段
实例化创建Bean实例对应于AbstractAutowireCapableBeanFactory的createBeanInstance方法。属性注入为实例化的Bean注入属性对应于populateBean方法。初始化执行Bean的初始化操作对应于initializeBean方法完成AOP代理等。
Spring使用三级缓存的策略如下
一级缓存singletonObjects存储已经完全初始化的单例Bean。二级缓存earlySingletonObjects存储早期的Bean对象未完全初始化时放入该缓存。三级缓存singletonFactories存储Bean工厂ObjectFactory用于创建Bean的早期引用。
缓存的工作流程如下
创建Bean实例Spring首先尝试从一级缓存singletonObjects中获取Bean如果没有则尝试从二级缓存earlySingletonObjects获取。如果依然没有找到则从三级缓存singletonFactories获取。提前曝光Bean当Spring检测到循环依赖时会将Bean的早期引用通过ObjectFactory创建的代理对象放入三级缓存。解决循环依赖当另一个Bean需要依赖尚未完全初始化的Bean时Spring会从三级缓存中获取其早期引用并将其放入二级缓存。完成初始化当Bean完全初始化后Spring会将其移至一级缓存确保Bean的正常使用。
图解分析对于通过构造器注入相互依赖的两个类A和BSpring的处理步骤如下
创建A时因A依赖BSpring将A的早期引用放入三级缓存。创建B时因B依赖ASpring从三级缓存中获取A的早期引用。B初始化完成后B的实例放入一级缓存。A随后也完成初始化并将其实例放入一级缓存。 三、为什么Spring使用三级缓存而不是二级缓存 代理对象的创建某些场景如AOP需要在Bean初始化的后期生成代理对象。如果仅使用二级缓存代理对象的创建可能会在Bean未完全初始化时进行导致代理不完整。三级缓存中的ObjectFactory可以确保在需要时动态生成代理对象。 延迟创建早期引用三级缓存允许Spring延迟创建早期引用从而在特殊场景下实现灵活的依赖处理避免了Bean在完全初始化前被错误引用。
三级缓存机制为Spring处理复杂的依赖关系提供了灵活性和可靠性同时保证了Bean初始化和代理生成的顺序。