Spring源码(五):Spring循环依赖详解

124 阅读2分钟

本文已参与「新人创作礼」活动,一起开启掘金创作。

一、单例@Autowired循环依赖(执行成功)

  1. 定义类CircularRefA和类CircularRefB。

image.png

image.png

假如先执行CircularRefA的实例化,首先调用getSingleton(beanName)从一、二、三级缓存中查找CircularRefA的bean,这时肯定是找不到的。

image.png 执行createBean(),doCreateBean()中createBeanInstance和polulateBean之间会将CircularRefA的bean添加到三级缓存。

image.png

polulateBean中通过@Autowired注解执行CircularRefB的实例化,也会执行CircularRefB的polulateBean,这时会再次执行CircularRefA的getBean,在三级缓存中可以找到CircularRefA的bean,完成CircularRefA的实例化。

image.png 接着执行CircularRefB的实例化,最后执行第一个CircularRefA的实例化。

二、单例构造方法上的循环依赖(抛出异常)

定义类CircularRefConA和类CircularRefConB。

image.png

image.png

假如还是CircularRefConA先实例化,getSingleton中把beanName添加到singletonsCurrentlyInCreation Set容器。然后在doCreateBean的createBeanInstance中会创建CircularRefConA的实例,CircularRefConB作为CircularRefConA构造方法的参数,会执行CircularRefConB的getBean。同理,作为CircularRefConB构造方法的参数,CircularRefConA的getBean也会执行。

image.png

image.png CircularRefConB构造方法的参数CircularRefConA执行getBean,一、二、三级缓存都没有,执行单例的getSingleton。beforeSingletonCreation中,用Set singletonsCurrentlyInCreation判断bean是否在创建中。add添加不成功,if条件为true,抛出异常。与第一种情况的区别是,没有把CircularRefConA的bean加到三级缓存。

image.png

image.png

三、多例@Autowired循环依赖(抛出异常)

定义类CircularRefPrototypeA和类CircularRefPrototypeB。

image.png

image.png CircularRefPrototypeA的bean在实例化过程中执行beforePrototypeCreation会被放入多例的缓存,缓存是TreadLocal类型的。

image.png

image.png CircularRefPrototypeA执行polulateBean会触发CircularRefPrototypeB的getBean,同理也会触发CircularRefPrototypeA的第二次getBean。doGetBean中,从多例的缓存中拿出beanName校验,返回true。

image.png if条件为true,抛出异常。

image.png ———————————————— 版权声明:本文为CSDN博主「自然醒zzz」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 原文链接:blog.csdn.net/JustPlayCod…