Spring源码解析(三)-Spring中循环依赖解决

@[TOC] spring Bean实例化----》 循环依赖
#在上一篇文章中,详细介绍了spring的实例化过程,但是没有介绍spring中循环依赖的解决方案,这边文章将在上一篇文章的基础上,重点讲解spring在实例化过程中是如何处理循环依赖的问题。

part1:循环依赖

我们首先来到实例化的入口,可以看到,只有单例,非懒加载,非抽象的才会提前实例化,多例天生就具有懒加载的特性,使用的时候才会进行实例化,但是实例化的时候也是调用getBean方法,所以这里主要看getBean方法具体的实现。
Spring源码解析(三)-Spring中循环依赖解决_第1张图片
来到getBean内部
第一步就是从缓存中获取(一级缓存),如果获取到直接就可以返回,如果获取不到会往下执行,
Spring源码解析(三)-Spring中循环依赖解决_第2张图片
来到下面首先会判断当前正在实例化的类如果scope 或者Prototype的作用域,,看是否已经在实例化,如果已经在实例化,直接回抛出异常,不让继续实例化(这个地方就是多例情况下循环依赖不能解决的,直接抛异常的直接原因,在这里就截断,不能往下继续实例化isPrototypeCurrentlyInCreation里面就是判断一个Set集合是否已经有正在实例化的对象)
/** Names of beans that are currently in creation. */
private final ThreadLocal prototypesCurrentlyInCreation =
new NamedThreadLocal<>(“Prototype beans currently in creation”);
Spring源码解析(三)-Spring中循环依赖解决_第3张图片
在这里插入图片描述
如果是单例,继续往下执行来到getSingleton地方,这个地方上一篇文章应该介绍过,这里再重点介绍一下,我们知道这里的createBean是一个回调函数,在getSingleton里面如果是当前对象需要实例化,就会调用到外面的createBean方法。我们先进入getSingleton里面。
Spring源码解析(三)-Spring中循环依赖解决_第4张图片
还是先从一级缓存中获取
Spring源码解析(三)-Spring中循环依赖解决_第5张图片
第一次实例化,肯定没有往下执行,来到beforeSingletonCreation,这个方法和多例里面那个是一样的功能,判断当前正在实例化的对象是否已经在Set集合中,如果已经存在说明是第二次来实例化,就会抛出异常,不让继续实例化。(阻断构造函数循环依赖的方案,所以构造函数循环依赖是解决不了的
private final Set inCreationCheckExclusions =
Collections.newSetFromMap(new ConcurrentHashMap<>(16));
在这里插入图片描述
Spring源码解析(三)-Spring中循环依赖解决_第6张图片
这里如果是属性注入的方式实例化,就会往下执行,来到singletonFactory.getObject(),这时候就会调用外层的createBean方法
Spring源码解析(三)-Spring中循环依赖解决_第7张图片
进入createBean,来到doCreateBean
Spring源码解析(三)-Spring中循环依赖解决_第8张图片
继续进入方法内在这里插入图片描述
进入createBeanInstance,来到获取有参数的构造函数的地方,这里就是获取有参数的构造函数,然后进行实例化,但是在有参构造函数实例化过程中,如果参数是依赖注入的化,就会先实例化参数的(会触发参数的实例化,去调用getBean操作),然后再实例化当前对象。
Spring源码解析(三)-Spring中循环依赖解决_第9张图片
如果是无参构造函数实例化,继续往下执行
在这里插入图片描述
实例化完成后,返回当前方法
Spring源码解析(三)-Spring中循环依赖解决_第10张图片
添加完三级缓存后,进行属性的依赖注入populateBean,这个时候,如果属性是引用类型,需要实例化的画,就会触发属性的getBean操作,继续实例化属性的实例,如果属性也引用了当前对象的话,在属性进行DI的时候,又会触发当前对象的getBean操作进行实例化,但是这个时候,当前对象进行第二次实例化的时候,因为会先从缓存中获取,由于三级缓存已经有内容,所以会直接返回,然后,属性的DI成功,属性实例化完成,属性实例化完成,当前对象也可以正常实例化,晚会返回。这就是spring在解决单例的无参构造函数的循环依赖的解决方案,这里文字描述会有点绕,下面会有图进行描述这个时序。
Spring源码解析(三)-Spring中循环依赖解决_第11张图片
接着往下执行,属性DI完成后,当前对象的实例化才可以说是真正完成,然后回到getSingleton的singletonFactory.getObject();这个地方,当前对象也就实例化完成
Spring源码解析(三)-Spring中循环依赖解决_第12张图片
然后就存入一级缓存,所以我们平时大部分时候获取对象,都是从一级缓存中获取
Spring源码解析(三)-Spring中循环依赖解决_第13张图片
针对上面的循环依赖,下面总结了了整个时序图
Spring源码解析(三)-Spring中循环依赖解决_第14张图片
由于图比较大,所以截图是缩小版,如果需要原版,请加关注后私信!!!
这篇文章是在上一篇文章spring实例化的基础上讲解spring的循环依赖解决方案:
总结一下:
1.单例的无参构造的循环依赖,spring是通过三级缓存来解决的
2.单例的参数循环依赖,spring是无法解决,直接回抛出异常
3.多例的循环依赖也是不支持的,直接抛出异常
后面还会持续更新spring相关源码讲解,敬请期待!!!

你可能感兴趣的:(spring,java)