Spring循环依赖

5 篇文章 0 订阅

什么是循环依赖?

当我们使用 IOC 进行依赖注入时,可能存在 A 依赖 B 且 B 依赖 A 的情况,如此将会形成 A->B->A 的循环依赖。
为了实例化 A,IOC 不得不先实例化 B,但是 B 的实例化又依赖 A,如此下去无穷无尽。那么循环依赖在 Spring 中的表现是怎样的呢?Spring 又是如何解决的?

案例

Spring 中依赖注入可以分为属性注入构造器注入两类,注入的 bean 也有单例多例的区分。我们可以对多种组合情况进行单独测试。

A单例&属性注入,B单例&属性注入

@Service
public class Test1 {

    @Autowired
    private Test2 test2;
}
@Service
public class Test2 {

    @Autowired
    private Test1 test1;
}

结果: 成功

A单例&属性注入,B单例&构造器注入

@Service
public class Test1 {

    @Autowired
    private Test2 test2;
}
@Service
public class Test2 {
    private Test1 test1;
    public Test2(Test1 test1) {
        this.test1 = test1;
    }
}

结果成功

A单例&构造器注入,B单例&属性注入

@Service
public class Test1 {

    private Test2 test2;
	public Test1(Test2 test2) {
        this.test2 = test2;
    }
}
@Service
public class Test2 {
    @Autowired
    private Test1 test1;
}

结果失败

The dependencies of some of the beans in the application context form a cycle:

┌─────┐
|  test1 defined in file [Test1.class]
↑     ↓
|  test2 (field private Test2.test1)
└─────┘

A单例&构造器注入,B单例&构造器注入

@Service
public class Test1 {

    private Test2 test2;
    
	public Test1(Test2 test2) {
        this.test2 = test2;
    }
}
@Service
public class Test2 {

    private Test1 test1;
    
    public Test2(Test1 test1) {
        this.test1 = test1;
    }
}

结果失败,日志同上

A多例模式&属性注入,B单例模式&属性注入

@Service
@Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
public class Test1 {

    @Autowired
    private Test2 test2;
}
@Service
public class Test2 {

    @Autowired
    private Test1 test1;
}

结果失败,日志同上

其他组合情况都会报相同错误,由于篇幅原因,本文就不再列举了。

结论

在循环依赖情况下,只有当优先加载的bean为 单例&属性注入 时,才能够成功注入。

为什么是这种现象?

分析getBean流程

图片稍小,建议放大了看

在这里插入图片描述
从这个简化的 getBean 流程中可以很清楚看到,Spring在获取单例时,会先从三个不同的缓存中查找bean,若找到便直接返回。若没有找到,通过createBean创建对象并立刻放入第三级缓存中。

若首先加载的bean为单例属性注入,则会先通过默认构造函数创建对象,并放入三级缓存中,后续再循环回来时,直接从缓存中获取对象,也就避免了死循环。

若首先加载的是构造函数注入,无法通过默认构造函数创建,只能使用指定的带参构造函数创建对象,故而在创建对象时需要提前获取参数依赖,这时bean并没创建成功,更没有在缓存中,出现循环时直接报错。

若首先加载的是多例,Spring不会保存对象的引用,更没有缓存。当再次循环回来时,发现相同的beanName已经在Set集合中,则报错。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Spring循环依赖指的是在Spring中,多个Bean之间存在相互依赖的情况。具体来说,当一个Bean A依赖于另一个Bean B,同时Bean B也依赖于Bean A时,就形成了循环依赖。这种情况下,Spring需要解决Bean的创建和依赖注入的顺序问题。 在Spring中,循环依赖问题是由于Bean的生命周期所引起的。Spring的Bean生命周期包括了Bean的实例化、属性注入、初始化以及销毁等过程。当出现循环依赖时,Spring会通过使用“提前暴露”的方式来解决这个问题。 具体来说,当Spring创建Bean A时,发现它依赖于Bean B,于是会创建一个A的半成品对象,并将其暂时放入一个缓存中。然后,Spring会继续创建Bean B,并将其注入到A的属性中。接着,Spring会继续完成B的创建,并将其放入缓存中。最后,Spring会将A的半成品对象交给B进行依赖注入,完成A的创建,并将其从缓存中移除。 需要注意的是,Spring循环依赖有一定的限制条件。例如,如果Bean A和Bean B都是单例模式,那么它们之间的循环依赖是无法解决的。因为单例模式下,Bean的创建和依赖注入是同时进行的,无法通过缓存来解决循环依赖。在这种情况下,程序员需要手动调整Bean的依赖关系或使用其他解决方案来避免循环依赖的问题。 综上所述,Spring循环依赖是指在Spring中多个Bean之间存在相互依赖的情况。Spring通过使用缓存和提前暴露的方式来解决循环依赖问题,但在某些情况下有一定的限制条件需要注意。<span class="em">1</span><span class="em">2</span><span class="em">3</span><span class="em">4</span>
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值