Spring 循环依赖
1. 什么是循环依赖?
抛开 Spring 框架不说,先来了解循环依赖的本质
从上图总可以看出依赖关系
这种依赖的关系就造成了循环依赖的发生
让我们从代码的角度来看看循环依赖是如何产生以及如何被解决的
定义两个类 A 和 B
public class A {
private B b;
}
public class B {
private A a;
}
模仿Spring:假装 A 和 B 是被 @Component 修饰,并且类中的字段假装是 @Autowired 修饰的,处理完放到 Map 中
/**
* 放置创建好的bean Map
*/
private static Map<String, Object> cacheMap = new HashMap<>(2);
public static void main(String[] args) {
// 假装扫描出来的对象
Class[] classes = {A.class, B.class};
// 假装项目初始化实例化所有bean
for (Class aClass : classes) {
getBean(aClass);
}
// check
System.out.println(getBean(B.class).getA() == getBean(A.class));
System.out.println(getBean(A.class).getB() == getBean(B.class));
}
@SneakyThrows
private static <T> T getBean(Class<T> beanClass) {
// 本文用类名小写 简单代替bean的命名规则
String beanName = beanClass.getSimpleName().toLowerCase();
// 如果已经是一个bean,则直接返回
if (cacheMap.containsKey(beanName)) {
return (T) cacheMap.get(beanName);
}
// 将对象本身实例化
Object object = beanClass.getDeclaredConstructor().newInstance();
// 放入缓存
cacheMap.put(beanName, object);
// 把所有字段当成需要注入的bean,创建并注入到当前bean中
Field[] fields = object.getClass().getDeclaredFields();
for (Field field : fields) {
field.setAccessible(true);
// 获取需要注入字段的class
Class<?> fieldClass = field.getType();
String fieldBeanName = fieldClass.getSimpleName().toLowerCase();
// 如果需要注入的bean,已经在缓存Map中,那么把缓存Map中的值注入到该field即可
// 如果缓存没有 继续创建
field.set(object, cacheMap.containsKey(fieldBeanName)
? cacheMap.get(fieldBeanName) : getBean(fieldClass));
}
// 属性填充完成,返回
return (T) object;
}
这段代码的效果,其实就是处理了循环依赖,并且处理完成后,cacheMap 中放的就是完整的 Bean
参考:https://mp.weixin.qq.com/s?__biz=MzAwNDA2OTM1Ng==&mid=2453143515&idx=1&sn=c15f40f4f45bf155fe7eddb7dc2c9080&scene=21