highlight: arduino-light
解决循环依赖\&bean实例化过程
什么是循环引用?
下面的代码中,两个类相互引用,就是循环依赖。
java
@Component
public class OrderService {
@Autowired
UserService userService;
public OrderService(){
System.out.println("OrderService——>"+userService);
}
}
@Component
public class UserService {
@Autowired
OrderService orderService;
public UserService(){
System.out.println("UserService->" + orderService);
}
}
/***
但是如果分别指定类的构造方法为以下形式 就无法解决。
因为通过构造方法传入要求必须传入一个实例化后的对象。
OrderService(UserService userService );
UserService(OrderService orderService);
***/
```java public class SpringTest {
public static void main(String[] args) {
AnnotationConfigApplicationContext context = new
AnnotationConfigApplicationContext();
context.register(Config.class);
context.refresh();
}
} ```
bean的实例化是在refresh()——>finishBeanFactoryInitialization(beanFactory);方法里完成的。
该方法只能实例化单例的类。
这两个类非常简单,就是相互引用了对方,也就是我们常常的说的循环依赖,spring是允许这样的循环依赖。
前提是1.单例的情况下的,2.非构造方法注入的情况下
spring默认是支持循序依赖的,但是仅仅是单例的类才可以。
上面代码从容器中能正常获取到bean,说明循环依赖成功。
但是spring的循环依赖其实是可以关闭的,spring提供了api来关闭循环依赖的功能。
spring也可以关闭循环依赖:
```java public class SpringTest {
public static void main(String[] args) {
AnnotationConfigApplicationContext context
= new AnnotationConfigApplicationContext();
//获取bean工厂
DefaultListableBeanFactory beanFactory =
(DefaultListableBeanFactory) context.getBeanFactory();
//关闭循环依赖
beanFactory.setAllowCircularReferences(false);
context.register(Config.class);
context.refresh();
System.out.println(context.getBean("orderService"));
}
} ```
运行上面的代码,就会报错
那么为什么setAllowCircularReferences(false);会关闭循环依赖呢?
首要明白spring的循环依赖是怎么做到的呢?spring源码当中是如何处理循环依赖的?
分析一下所谓的循环依赖其实无非就是属性注入,或者就是大家常常说的自动注入, 故而搞明白循环依赖就需要去研究spring自动注入的源码。
Spring的属性注入属于spring bean的生命周期一部分;怎么理解spring bean的生命周期呢?
注意笔者这里并不打算对bean的生命周期大书特书,只是需要读者理解生命周期的概念,细节下一篇文章再写。
解决循环依赖的2个前提
单例
非构造方法注入
Spring循环依赖不能解决的问题
不支持多例
不支持构造方法注入的bean
如何解决循环依赖
1.\@Autowired自动注入
```java @Component public class A { @Autowired B b; }
@Component public class B { @Autowired A a; } ```
2.set方法注入
```java @Component public class A1 { B1 b; @Autowired public void setB1(B1 b) { this.b = b; } }
@Component public class B1 { A1 a; @Autowired public void setA(A1 a) { this.a = a; } } ```
3.构造方法循环依赖@Lazy
java
@Component
public class C {
D d;
public C(@Lazy D d) {
this.d = d;
}
}
@Component
public class D {
C c;
public D(@Lazy C c) {
this.c = c;
}
}