Spring的循环依赖问题


循环依赖

要了解循环依赖,我们首先要知道代理的一些知识

关于代理

在Spring中我们可以使用ProxyFactory来实现代理的增强


aop的底层是靠的是动态代理,一个切面中可能有多个方法,用来增强,在实例中一个方法被解析为一个advisor切面

代理创建实际:用到aop的地方,bean的后处理器 看是否有自定义targetSource

产生循环依赖调用 getEarlyBeanReference 提前创建代理

循环依赖

首先要了解什么是循环依赖

@Component
public class A {
 
    @Autowired
    B b;
}
 
@Component
public class B {
 
    @Autowired
    A a;
}

在对象A创建过程中,需要注入B,因为容器中没有B,则去创建B,B创建过程中又需要注入A,而A在等待B的创建,B在等待A的创建,导致两者都无法创建成功,无法加入到单例池供用户使用。

//三级缓存
public class DefaultSingletonBeanRegistry extends SimpleAliasRegistry implements SingletonBeanRegistry {
 
    //一级缓存(单例池,经过完成生命周期的对象会放入其中)
    private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);
 
    //二级缓存(刚实例化还未初始化的原始对象会放入其中)
    private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);
    
    //三级缓存(存放创建某个对象的工厂)
    private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);

一级缓存

保证单例bean只创建一次


一个对象创建,依赖注入,初始化完成后才能放到一级缓存

二级缓存


为了解决循坏依赖中代理类创建过晚的问题

三级缓存


首先从缓存中得到A,A此时为空,自己创建,实例化后在二级缓存中放入这个实例的引用放在singletonFactories中(此时是半成品),然后依赖注入发现需要B,先去二级缓存中找B,没找到,这时去实例化B,B实例化后放入二级缓存,然后去依赖注入A,完成初始化后,二级缓存B移入一级缓存。A再从一级缓存拿到B的实例

如果有代理对象呢?

如果没有使用到代理时,使用set注入,一级缓存和三级缓存已经解决了循环依赖的问题。比如A依赖B,实例化A进行依赖注入时,发现需要B,于是去实例化B,实例化后依赖注入时需要A,这时A已经存在了引用,只不过A此时功能不全,没有初始化。

二级缓存主要是为了解决循环依赖代理的问题。

很明白,我们只需要让代理类创建时机提前就可以解决。

1.

首先左上角的fa->pa||a 可以看做一个工厂方法,它可以判断我们是否需要代理,如果需要代理会给我们提前创建代理对象返回,如果不需要就正常创建

走到蓝色部分是 factories.put(fa)放入到singletonFactories中,也就是三级缓存中。

2.

到B的创建部分, factories.get需要A的对象实例,这是工厂中的fa会返回一个代理对象,并将返回的代理对象放入二级缓存中

3.

debug一下

有代理情况

如果是构造注入发生循坏依赖呢?

解决办法:
1.推迟B对象的获取

代理解决

objectFactory解决

二级缓存只有在循环依赖时才会存东西



Set注入和构造注入


在A中有一个setB方法用来接收B对象的实例。那么Spring实例化A对象的过程如下

在不考虑Bean的初始化方法和一些Spring回调的情况下,Spring首先去调用A对象的构造函数实例化A,然后查找A依赖的对象本例子中是B(合作者)。一但找到合作者,Spring就会调用合作者(B)的构造函数实例化B。如果B还有依赖的对象Spring会把B上依赖的所有对象都按照相同的机制实例化然后调用A对象的setB(B b)把b对象注入给A。因为Spring调用一个对象的set方法注入前,这个对象必须先被实例化。所以在”使用set方法注入”的情况下Spring会首先调用对象的构造函数。

构造注入的过程

如果发现配置了对象的构造注入,那么Spring会在调用构造函数前把构造函数需要的依赖对象都实例化好,然后再把这些实例化后的对象作为参数去调用构造函数。

/**
在使用构造函数和set方法依赖注入时,Spring处理对象和对象依赖的对象的顺序时不一样的。一般把一个Bean设计为构造函数接收依赖对象时,其实是表达了这样一种关系:他们(依赖对象)不存在时我也不存在,即“没有他们就没有我”。

通过构造函数的注入方式其实表达了2个对象间的一种强的聚合关系:组合关系。就比如一辆车如果没有轮子、引擎等部件那么车也就不存在了。而且车是由若干重 要部件组成的,在这些部件没有的情况下车也不可能存在。这里车和他的重要部件就时组合的关系。如果你的应用中有这样类似的场景那么你应该使用“构造函数注 入”的方式管理他们的关系。“构造函数注入”可以保证合作者先创建,在后在创建自己。

通过set方法注入的方式表达了2个对象间较弱的依赖关系:聚合关系。就像一辆车,如果没有车内音像车也时可以工作的。当你不要求合作者于自己被创建 时,“set方法注入”注入比较合适。

虽然在理论上“构造函数注入”和“set方法注入”代表2种不同的依赖强度,但是在spring中,spring并不会把无效的合作者传递给一个 bean。如果合作者无效或不存在spring会抛出异常,这样spring保证一个对象的合作者都是可用的。所以在spring中,“构造函数注入”和 “set方法注入”唯一的区别在于2种方式创建合作者的顺序不同。
使用构造函数依赖注入时,Spring保证所有一个对象所有依赖的对象先实例化后,才实例化这个对象。(没有他们就没有我原则)

使用set方法依赖注入时,Spring首先实例化对象,然后才实例化所有依赖的对象

文章作者: 蛰伏
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 蛰伏 !
  目录