Spring源码入门(一)之-----循环依赖

Spring是如何解决循环依赖问题的?这个问题算是关于Spring的一个高频面试题,如果不刻意研读,即使度过源码,面试者也不一定能够一下子回答得上。
什么是循环依赖?
循环依赖其实就是对象之间的循环引用,即两个或两个以上的Bean互相持有对方,最终形成闭环。
在这里插入图片描述

Spring 循环依赖有几种方式?
1.原型模式循环依赖【无法解决】 如果是单列的话,就不会报错
Spring默认的单列(Singleton)的场景是支持循环依赖的,不报错
原型(Prototype)的场景是不支持循环依赖的,会报错
2.单列Bean循环依赖–构造参数产生依赖【无法解决】
3.单列Bean循环依赖-setter产生依赖【可以解决】

同时,循环依赖会报错: BeanCurrentlyInCreationException

Spring内部如何解决循环依赖的问题呢?
三级缓存,分别为:
一级缓存(单列池),存放一级经历完整生命周期的Bean对象,singletonObjects(ConcurrentHashMap)
二级缓存,存放早起暴露的Bean对象,Bean的生命周期未结束,bean的属性还未填充,也就是实例化但是未初始化的bean放到该缓存中,earlySingletonObjects
三级缓存,存放可以生成Bean的工厂,singletonFactories

只有单列的bean会通过三级缓存提前暴露来解决循环依赖问题,而非单列的bean,每次从容器中获取都是一个新的对象,都会被创建,所以非单列的bean是没有缓存的,不会将其放到三级缓存中。

在解决循环依赖的过程中,Spring会用到四大方法。
1.getSingleton();
通过bean的名称,去一级缓存查询;二级缓存查询,三级缓存查询;
2.doCreateBean();
创建bean
3.populateBean();
属性填充
4.addSingleton();
放到

在这里插入图片描述

Bean创建的流程如下:
1.A创建的过程中需要B,于是A将自己放到三级缓存中,去实例化B

2.B实例化的时候发现需要A,于是B先查一级缓存,没有,再去查二级缓存,还是没有,再去查三级缓存,找到了A,然后把三级缓存的里面的这个A放到二级缓存,并删除三级缓存里面的A

3.B 顺利初始化完毕后,将自己放到一级缓存中(此时B里面的A依然是创建中的状态),然后回来接着创建A,此时B已经创建结束,直接从一级缓存中拿到了B,然后完成创建,并将A自己放入到一级缓存里面。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

virtuousOne

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值