再探循环依赖 → Spring 是如何判定原型循环依赖和构造方法循环依赖的?
开心一刻
一天,侄子和我哥聊天,我坐在旁边听着
侄子:爸爸,你爱我妈妈吗?
哥:这话说的,不爱能有你吗?
侄子:确定有我不是因为荷尔蒙吗?
哥:因为什么荷尔蒙,因为爱情!
侄子:那我妈花点钱,你咋老说呢?
哥:这你就不懂了,挣钱本不易,花钱要仔细
侄子:快得了吧,挣钱这么少,我妈都没跑,给你照顾家,钱还不让花
哥:我发现你这孩子怎么不知道好赖呢,我攒钱不是为了给你去媳妇啊
侄子:那你赶紧给我妈花吧,我妈要是跑了,你还得花钱娶一个,到最后,钱我捞不着,亲妈还混没了
我:通透!!!
写在前面
Spring 中常见的循环依赖有 3 种:单例 setter 循环依赖、单例构造方法循环依赖、原型循环依赖
关于单例 setter 循环依赖,Spring 是如何甄别和处理的,可查看:Spring 的循环依赖,源码详细分析 → 真的非要三级缓存吗
单例构造方法循环依赖
何谓单例构造方法循环依赖了,我们看具体代码就明白了
两个要素:① scope 是默认值,也就是 singleton;② 多个实例之间通过构造方法形成了循环依赖
这种情况下,Spring 是怎么处理的了,我们先来看看执行结果
Spring 启动过程中报错了: Error creating bean with name 'cat': Requested bean is currently in creation: Is there an unresolvable circular reference?
问题就来了:Spring 是如何甄别单例情况下的构造方法循环依赖的,然后进行报错的
大家先把这个问题暂留在心里,我们再来看看什么是原型循环依赖
原型循环依赖
同样,我们直接看代码就明白何谓原型循环依赖了
同样是 2 个要素:① scope 不是默认值,而是 prototype,也就是原型,每次获取该实例的时候都会新建;② setter 循环依赖
这种情况下 Spring 又会有什么样的执行结果了
Spring 启动正常,但从 Spring 容器获取 loop 实例的时候,报了同样的错误
问题来了:① Spring 是如何甄别原型循环依赖的,然后进行报错提示的
② 为什么两种情况的报错时机会不一致,一个在 Spring 启动过程中,一个却在使用 Spring 的过程中
示例代码地址:spring-circle-dependence-type
上面的 3 个问题,概括下就是
1、Spring 是如何甄别单例情况下的构造方法循环依赖的
2、Spring 是如何甄别原型循环依赖的
3、为什么单例构造方法循环依赖和原型循环依赖的报错时机不一致
我们慢慢往下看,跟源码的过程可能比较快,大家看仔细了
还是那句话
看完之后仍有疑问,可以评论区留言,也可以自行去查阅相关资料进行解疑
源码起点
Spring 读取和解析 xml 的过程,我们就不去跟了,我们重点跟一下我们关注的内容
我们从 DefaultListableBeanFactory 类的 preInstantiateSingletons 方法作为起点
按如下顺序可以快速的找到起点,后面两种情况都从此处开始进行源码跟踪
构造方法循环依赖的甄别
闲话少说,我们直接开始跟源码
获取 cat 实例
cat 的 RootBeanDefinition 中有几个属性值得我们注意下
接着往下走
我们来到了 createBeanInstance 方法,此时 Set<String> singletonsCurrentlyInCreation 只存放了 cat
singletonsCurrentlyInCreation 看字面意思就知道,存放的是当前正在创建中的单例对象名
我们接着往下跟
由于 constructorArgumentValues 中有元素,所以需要通过有参构造函数来创建 cat 对象
因为构造函数的参数是 Dog 类型的 dog ,所以通过反射调用 Cat 的有参构造函数来创建 cat 之前,需要先从 Spring 容器中获取到 dog 对象
获取 Cat 构造函数依赖的 dog 实例
所以流程又来到了我们熟悉的 getBean ,只是现在获取的是 dog ;获取流程与获取 cat 时一样,所以跟的速度会快一些,大家注意看我停顿的地方
此时 singletonsCurrentlyInCreation 存放了 cat 和 dog ,表示他们都在创建中
又来到了 createBeanInstance ,过程与之前 cat 的过程一样,我们接着往下看
又来到了熟悉的 getBean ,需要从 Spring 容器获取 Dog 构造函数依赖的 cat 对象
获取 Dog 构造函数依赖的 cat 对象
接下来重点来了,大家看清楚了
因为 singletonsCurrentlyInCreation 已经存在 cat 了, !this.singletonsCurrentlyInCreation.add(beanName) 结果就是 true
说明陷入死循环了,所以抛出了 BeanCurrentlyInCreationException
我们在控制台看到的异常信息就从这来的
原型循环依赖的甄别
原型类型的实例有个特点:每次获取都会重新创建一个实例,那在 Spring 启动过程中,还有创建的必要吗?
Spring 启动不创建 prototype 类型的实例
我们来跟下源码就明白了
关键代码
不符合上述 3 个条件的实例,在 Spring 启动过程中都不会被创建
下面接着讲正题,来看看 Spring 是如何甄别原型循环依赖的
获取 loop 实例
在 loop 实例创建之前,调用了 beforePrototypeCreation 方法,将 loop 名放到了 ThreadLocal<Object> prototypesCurrentlyInCreation
表示当前线程正在创建 loop ,我们接着往下看
原型类型的对象创建过程分两步:① 实例化(反射调构造方法),② 初始化(属性填充),和单例类型对象的创建过程是一样的
依赖的处理是在初始化过程中进行的, loop 对象依赖 circle 属性,所以对 loop 对象的 circle 属性进行填充的时候,需要去 Spring 容器获取 circle 实例
又来到了我们熟悉的 getBean ,获取 loop 依赖的 circle 实例,我们继续往下跟
在 circle 对象创建之前,同样调用了 beforePrototypeCreation 方法,那么此时 prototypesCurrentlyInCreation 中就同时存在 loop 和 circle
表示当前线程正在创建 loop 实例和 circle 实例;继续往下走
兜兜转转又来到了 getBean ,获取 circle 对象依赖的 loop 属性,接下来是重点,大家看仔细了
因为 prototypesCurrentlyInCreation 中存在 loop 了,说明当前线程正在创建 loop 实例
而现在又要创建新的 loop ,说明陷入死循环了,所以抛出了 BeanCurrentlyInCreationException
总结
经过上面的梳理,相信大家对之前的三个问题都没有疑问了,我们来总结下
1、Spring 是如何甄别单例情况下的构造方法循环依赖的
Spring 通过 Set<String> singletonsCurrentlyInCreation 记录当前正在创建中的实例名称
创建实例对象之前,会判断 singletonsCurrentlyInCreation 中是否存在该实例的名称,如果存在则表示死循环了,那么抛出 BeanCurrentlyInCreationException
2、Spring 是如何甄别原型循环依赖的
Spring 通过 ThreadLocal<Object> prototypesCurrentlyInCreation 记录当前线程正在创建中的原型实例名称
创建原型实例之前,会判断 prototypesCurrentlyInCreation 中是否存在该实例的名称,如果存在则表示死循环了,那么抛出 BeanCurrentlyInCreationException
3、为什么单例构造方法循环依赖和原型循环依赖的报错时机不一致
单例构造方法实例的创建是在 Spring 启动过程中完成的,而原型实例是在获取的时候创建的
所以两者的循环依赖的报错时机不一致
参考
Spring 的循环依赖,源码详细分析 → 真的非要三级缓存吗
再探循环依赖 → Spring 是如何判定原型循环依赖和构造方法循环依赖的?的更多相关文章
- 再探java基础——break和continue的用法
再探java基础——break和continue的用法 break break可用于循环和switch...case...语句中. 用于switch...case中: 执行完满足case条件的内容内后 ...
- Spring 循环依赖的三种方式(三级缓存解决Set循环依赖问题)
本篇文章解决以下问题: [1] . Spring循环依赖指的是什么? [2] . Spring能解决哪种情况的循环依赖?不能解决哪种情况? [3] . Spring能解决的循环依赖原理(三级缓存) 一 ...
- Spring源码学习笔记9——构造器注入及其循环依赖
Spring源码学习笔记9--构造器注入及其循环依赖 一丶前言 前面我们分析了spring基于字段的和基于set方法注入的原理,但是没有分析第二常用的注入方式(构造器注入)(第一常用字段注入),并且在 ...
- 【spring源码系列】之【Bean的循环依赖】
希望之光永远向着目标清晰的人敞开. 1. 循环依赖概述 循环依赖通俗讲就是循环引用,指两个或两个以上对象的bean相互引用对方,A依赖于B,B依赖于A,最终形成一个闭环. Spring循环依赖的场景有 ...
- 再探ASP.NET 5(转载)
就在最近一段时间,微软又有大动作了,在IDE方面除了给我们发布了Viausl Studio 2013 社区版还发布了全新的Visual Studio 2015 Preview. Visual Stud ...
- spring揭秘 读书笔记 二 BeanFactory的对象注册与依赖绑定
本文是王福强所著<<spring揭秘>>一书的读书笔记 我们前面就说过,Spring的IoC容器时一个IoC Service Provider,而且IoC Service Pr ...
- 第四节:SignalR灵魂所在Hub模型及再探聊天室样例
一. 整体介绍 本节:开始介绍SignalR另外一种通讯模型Hub(中心模型,或者叫集线器模型),它是一种RPC模式,允许客户端和服务器端各自自定义方法并且相互调用,对开发者来说相当友好. 该节包括的 ...
- c++再探string之eager-copy、COW和SSO方案
在牛客网上看到一题字符串拷贝相关的题目,深入挖掘了下才发现原来C++中string的实现还是有好几种优化方法的. 原始题目是这样的: 关于代码输出正确的结果是()(Linux g++ 环境下编译运行) ...
- 码农小汪-spring框架学习之2-spring IoC and Beans 控制反转 依赖注入 ApplicationContext BeanFactory
spring Ioc依赖注入控制反转 事实上这个东西很好理解的,并非那么的复杂. 当某个Java对象,须要调用还有一个Java对象的时候(被依赖的对象)的方法时.曾经我们的做法是怎么做呢?主动的去创建 ...
随机推荐
- String的20个方法
String的20个方法 面试题 1.new和不new的区别 String A="OK"; String B="OK";//会去常量池查找有没有"Ok ...
- Leetcode(18)-四数之和
给定一个包含 n 个整数的数组 nums 和一个目标值 target,判断 nums 中是否存在四个元素 a,b,c 和 d ,使得 a + b + c + d 的值与 target 相等?找出所有满 ...
- Bootstrap导航组件
Bootstrap 中的导航组件都依赖同一个 .nav 类,状态类也是共用的.改变修饰类可以改变样式. 标签页 注意 .nav-tabs 类依赖 .nav 基类 <ul class=" ...
- POJ 2923 Relocation(状压DP)题解
题意:有2辆车运货,每次同时出发,n(<10),各自装货容量c1 c2,问最少运几次运完. 思路:n比较小,打表打出所有能运的组合方式,用背包求出是否能一次运走.然后状压DP运的顺序. 代码: ...
- Linux 驱动框架---dm9000分析
前面学习了下Linux下的网络设备驱动程序的框架inux 驱动框架---net驱动框架,感觉知道了一个机器的大致结构还是不太清楚具体的细节处是怎么处理的,所以今天就来以dm9000这个网上教程最多的驱 ...
- STM32 单片机的USART的奇偶校验 误区(坑)
当STM32的串口配置成带有奇偶校验位的情况下,需要软件校验是否发生奇偶校验错误,硬件只是置起奇偶校验错误标志位,并将错误的数据放到DR寄存器中,同时置起RXEN标志位,如果使能中断还是会正常进入中断 ...
- uni-app 实战-打包 📦 App
uni-app 实战-打包 App Android & iOS 证书 广告 refs xgqfrms 2012-2020 www.cnblogs.com 发布文章使用:只允许注册用户才可以访问 ...
- React Native & Security
React Native & Security https://reactnative.dev/docs/security React Native blogs https://reactna ...
- Publish/Subscribe Pattern & Vanilla JavaScript
Publish/Subscribe Pattern & Vanilla JavaScript https://en.wikipedia.org/wiki/Publish–subscribe_p ...
- webpack 4 & dev server
webpack 4 & dev server proxy https://webpack.js.org/configuration/dev-server/#devserverproxy htt ...