栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

Spring如何解决循环依赖

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

Spring如何解决循环依赖

首先要先了解下Spring Bean的生命周期

例如出现这样的循环依赖

@Component
public Class A { 
  @Autowired private B b;
}

@Component
public Class B { 
  @Autowired private A b;
}

A依赖B的情况加载流程

 A、B相互依赖的加载过程

 以上就会出现一个问题,由于 a、b 都是单例 Bean,加载 b 的时候,到了上图中标红的阶段后,b 依赖注入的 a 的引用应该是通过 getBean(A) 得到的引入,这个时候A还没有放入单例池中,又再一次走入了 A 的创建逻辑,此时就是发生了循环依赖。

一级缓存:单例池 singletonObjects
private final Map> singletonFactories = new HashMap<>(16);

我们都知道如果是单例的 Bean,每次 getBean(beanName)返回同一个 bean,也就是在整个 ApplicationContext 里面,仅有一个单例 Bean,单例 Bean 创建完成后就放在 singletonObjects 这个 Map 里面,这就是一级缓存。此时说的“创建完成”指的是图一的第 6 步骤,图三中 getBean("B") 的过程中,a 是没有加入到一级缓存中,所以在 getBean("B") 的流程中,b 依赖了 a,此时 b 是找不到 a 对象的。依然会无法解决循环引用的问题。

二级缓存:earlySingletonObjects
private final Map earlySingletonObjects = new ConcurrentHashMap<>(16);

这个时候我们考虑再引入一个 Map 存放引用,earlySingletonObjects 这个 map 我们打算存放提前暴露 bean 的引用,实例化以后,我们就把对象放入到 earlySingletonObjects 这个 map 中,这样在 加载 b 的过程中,b.setA(getBean("a")),我们就可以在 earlySingletonObjects 拿到 a 的引用,此时 a 仅仅经过了实例化,并没有设置属性。流程如下:

 到目前为止,发现使用二级缓存似乎就能解决我们的问题。看起来很美好,这是 Spring IOC 的特性,Spring 的另一大特性是 AOP 面向切面编程,动态增强对象,不管使用 JDK 的动态代理和 Cglib 动态代理,都会生成一个全新的对象。下图中我标出了 AOP 动态增强的位置。

此时就会出现一个问题,因为经过 AOP 以后,生成的是增强后的 bean 对象,也就是一个全新的对象,我们可以看到经过图中的流程后,单例池中会存在两个 bean:增强后的 a、b 对象,此时 a 对象中依赖的 b 为增强后的,而 b 对象依赖的 a 是为原始对象,未增强的,单例池中只存在增强后的a对象。所以使用二级缓存解决不了循环依赖中发生过 aop 的引用问题。 

三级缓存:singletonFactories
private final Map> singletonFactories = new HashMap<>(16);

为了解决二级缓存中 AOP 生成新对象的问题,Spring 中的解决方案是:提前 AOP,如果我们能够提前 AOP 就能解决上面的问题了,提前 AOP 指的就是,在 加载 B 的流程中,如果发生了循环依赖,就是说 b 又依赖了 a,我们就要对 a 执行 aop,提前获取增强以后的 a 对象,这样 b 对象依赖的 a 对象就是增强以后的 a 了。三级缓存的 key 是 beanName,value 是一个 lambda 表达式,这个 lambda 表达式的作用就是进行提前 AOP。


上面就是三级缓存的作用,其中有个三级缓存到二级缓存的升级过程,这个非常重重要,这个主要是防止重复 aop。 

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/331916.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号