当前位置: 首页 > news >正文

人人都能看懂的Spring源码解析,Spring如何解决循环依赖

人人都能看懂的Spring源码解析,Spring如何解决循环依赖

  • 原理解析
    • 什么是循环依赖
    • 循环依赖会有什么问题?
    • 如何解决循环依赖
      • 问题的根本原因
      • 如何解决
    • 为什么需要三级缓存?
    • Spring的三级缓存
  • 源码走读
    • Spring的三级缓存
    • 提前暴露
    • getSingleton方法
  • 总结

往期内容:

  1. 人人都能看懂的Spring底层原理,看完绝对不会懵逼
  2. 简单易懂的Spring扩展点详细解析,看不懂你来打我
  3. 人人都能看懂的Spring源码解析,配置解析与BeanDefinition加载注册
  4. 简单易懂又非常牛逼的Spring源码解析,ConfigurationClassPostProcessor的具体逻辑
  5. 简单易懂值得收藏的Spring源码解析,依赖注入和bean的初始化

原理解析

什么是循环依赖

两个Bean,BeanA和BeanB,它们都有一个引用指向对方,这就是最简单的循环依赖。还有更复杂的循环依赖,涉及到多个bean,只要其中形成环,就是循环依赖。

在这里插入图片描述

循环依赖会有什么问题?

如果Spring不对这种情况做处理,那么在进行依赖注入的时候就会出现死循环

在这里插入图片描述

如何解决循环依赖

问题的根本原因

要解决循环依赖的问题,首先就要分析导致这个问题的根本原因。

因为上面是假设Spring只有一个缓存,那就是单例缓存池,而且该缓存池是用于存放已初始化完成的bean,而没有完成依赖注入的bean不算是初始化完成的bean,所以不会放入单例缓存池中。

所以beanA依赖beanB时,只能实例化一个beanB,然后对其进行依赖注入,而beanB又依赖了beanA,所以又只能实例化一个beanA,对其进行依赖注入,这样就没完没了了。

如何解决

解决办法就是多增加一个缓存(二级缓存),用于存放已经实例化但是未完成初始化的bean。因为增加了一个二级缓存,在bean被实例化的之后就预先放入到该二级缓存中,那么在后续其他bean进行依赖注入的时候,发现依赖了该bean,然后检查到二级缓存中有,就不需要往下进行实例化。

在这里插入图片描述

比如上图,在第二次getBean(beanA)的时候,二级缓存中已经存放了beanA,因此不会往下再次进行beanA的实例化,而是直接从缓存中取,死循环因此就解决了。

为什么需要三级缓存?

那看起来搞两级缓存就能解决问题,但是Spring实际上是有三级缓存的,那为什么Spring需要搞三级缓存呢

那是因为Spring要处理AOP,我们可以想一想,如果只有两级缓存,要怎么同时处理循环依赖和AOP呢?

此时就要在每个bean实例化后,放入二级缓存之前,都要判断一下这个bean是否需要进行AOP处理:如果需要,就要先通过动态代理生成代理对象,放入二级缓存的就是代理对象;如果不需要进行AOP处理,那么就把原始对象放入二级缓存。

这个显然不符合Spring的设计。Spring的设计是把AOP放到依赖注入完成以后,在初始化阶段触发的,而现在却要把AOP的处理提前到实例化之后。

那如果还是要把AOP放到初始化阶段触发,只有涉及到循环依赖的bean才提前进行AOP呢?那就要在每次从二级缓存获取bean的时候,都要判断一下是否需要进行AOP处理,这样就太繁琐了。

因此最好的解决办法就是再加一级缓存,三级缓存。先从二级缓存中取,如果有,直接返回,如果二级缓存没有,再从三级缓存中拿,放入二级缓存,并从三级缓存删除,并且只有从三级缓存中取的时候,才判断是否需要进行AOP处理

在这里插入图片描述

Spring的三级缓存

在这里插入图片描述

  • singletonObjects:一级缓存,Map<String, Object>类型,key是beanName,value是已初始化完成的bean。
  • earlySingletonObjects:二级缓存,Map<String, Object>类型,key是beanName,value是已实例化完成但是未进行初始化的bean。
  • singletonFactories:三级缓存,Map<String, ObjectFactory<?>>,key是beanName,value是ObjectFactory<?>类型。ObjectFactory是一个函数式接口,可以视为一个回调函数,里面的逻辑就是调用bean后置处理判断是否需要进行AOP处理,如果需要则进行AOP处理,返回一个代理对象,如果不需要进行AOP处理,则返回原始对象。

源码走读

Spring的三级缓存

在这里插入图片描述

上面描述的三级缓存,就是DefaultSingletonBeanRegistry的成员变量

在这里插入图片描述

DefaultSingletonBeanRegistry又是DefaultListableBeanFactory的祖先类,所以三级缓存就在DefaultListableBeanFactory里面。

提前暴露

在这里插入图片描述
AbstractAutowireCapableBeanFactory#doCreateBean方法,首先调用createBeanInstance方法进行bean的实例化

在这里插入图片描述
实例化以后,就是判断是否需要提前暴露(earlySingletonExposure为true),如果需要,则调用addSingletonFactory方法进行提前暴露

然后下面的populateBean方法里面就是依赖注入的处理,initializeBean方法里面就是bean初始化的处理,这两个方法在前面的文章已经讲过。

需要提前暴露的条件:

  • mbd.isSingleton():是否单例。
  • this.allowCircularReferences:是否允许循环依赖,默认为true(但是高版本改成默认为false了,Spring希望我们把代码写的更好,而不是指望Spring来给我们处理循环依赖的问题)。
  • isSingletonCurrentlyInCreation(beanName):该bean是否正在创建中。

三个条件都满足,就调用addSingletonFactory方法,提前暴露。

在这里插入图片描述
addSingletonFactory方法里面,就是把beanName和ObjectFactory类型的对象singletonFactory,放入三级缓存singletonFactories。这里的singletonFactory的就是外面的lambda表达式 () -> getEarlyBeanReference(beanName, mbd, bean)。ObjectFactory是一个函数式接口,里面有一个getObject方法,而singletonFactory的getObject方法就是调用getEarlyBeanReference(beanName, mbd, bean)。

getSingleton方法

一个bean在依赖注入阶段,要处理对别的bean的依赖,会调用beanFactory的getBean方法,尝试从容器中取,getBean方法会先调用getSingleton方法,尝试从三级缓存中取

在这里插入图片描述

  1. 首先从第一级缓存取this.singletonObjects.get(beanName),不是null就返回
  2. 如果第一级缓存取到的是null,则从第二级缓存取this.earlySingletonObjects.get(beanName),不是null就返回
  3. 如果第二级缓存取到的还是null,则从第三级缓存取this.singletonFactories.get(beanName)
  4. 如果第三级缓存取到的不是null,则调用getObject方法,获取getObject方法返回的对象,放入二级缓存,从三级缓存中删除
  5. 如果第三级缓存取到的还是null,就要走创建bean的逻辑

这里的singletonFactory的getObject实现方法,就是addSingletonFactory方法的第二个参数的lambda表达式 () -> getEarlyBeanReference(beanName, mbd, bean),也就是会调用 getEarlyBeanReference(beanName, mbd, bean) 方法获取返回的对象。
在这里插入图片描述

可以看到遍历所有的bean后置处理器,判断如果是SmartInstantiationAwareBeanPostProcessor类型,就回调getEarlyBeanReference方法,这里会进入到AbstractAutoProxyCreator#getEarlyBeanReference方法。

在这里插入图片描述
wrapIfNecessary方法就是判断如果当前bean需要进行AOP处理,则进行动态代理,返回代理对象;如果不需要进行AOP处理,会返回原始对象。

总结

以上就是关于“Spring如何处理循环依赖”的讲解的全部内容。下面做一个简单总结:

  • Spring是通过三级缓存解决循环依赖的。一级缓存是singletonObjects;二级缓存是earlySingletonObjects;三级缓存是singletonFactories。
  • 一个bean在实例化完成后,在进行依赖注入之前,如果判断满足三个条件(当前bean是一个单例bean,并且允许循环依赖,并且当前bean正在创建中),就会进行提前暴露,放入到第三级缓存中。
  • 在依赖注入阶段,会调用beanFactory的getBean方法获取依赖的bean,getBean方法会先调用getSingleton方法尝试从三级缓存中取。
  • getSingleton方法先尝试从第一级缓存取,如果没有再从第二级缓存取,如果再没有就从第三级缓存取,如果第三级缓存中有,会调用getObject方法获取返回对象,再放入第二级缓存,从第三级缓存中删除。
  • singletonFactory的getObject方法,会调用getEarlyBeanReference方法,里面会调用到AbstractAutoProxyCreator的getEarlyBeanReference方法,判断是否要进行AOP处理,如果需要会进行动态代理返回代理对象,如果不需要,就返回原始对象。

相关文章:

人人都能看懂的Spring源码解析,Spring如何解决循环依赖

人人都能看懂的Spring源码解析&#xff0c;Spring如何解决循环依赖原理解析什么是循环依赖循环依赖会有什么问题&#xff1f;如何解决循环依赖问题的根本原因如何解决为什么需要三级缓存&#xff1f;Spring的三级缓存源码走读Spring的三级缓存提前暴露getSingleton方法总结往期…...

Linux上搭建Discuz论坛

一.准备工作 1.下载php*&#xff0c;mariadb-server 2.上传Discuz3.5压缩包并解压 二.搭建过程 基于redhat 9 版本和Discuz3.5&#xff0c;php8.0&#xff0c;mariadb10.5演示 一.准备工作 1.下载php*&#xff0c;mariadb-server [rootredhat9 aaa]# yum install -y php*…...

【蓝桥杯专题】 树状数组(C++ | 洛谷 | acwing | 蓝桥)

菜狗现在才开始备战蓝桥杯QAQ 文章目录【蓝桥杯专题】 &#xff08;C | 洛谷 | acwing | 蓝桥&#xff09;什么是线段数组??1264. 动态求连续区间和数星星线段树AcWing 1270. 数列区间最大值PPPPPPP【蓝桥杯专题】 &#xff08;C | 洛谷 | acwing | 蓝桥&#xff09; 什么是…...

QCefView编译配置(Windows-MSVC)(11)

QCefView编译配置&#xff08;Windows-MSVC&#xff09; 文章目录QCefView编译配置&#xff08;Windows-MSVC&#xff09;1、概述2、准备工作3、添加环境变量4、更换cef源码版本5、CMake构建6、Visual Studio编译7、安装编译后的文件8、验证编译结果更多精彩内容&#x1f449;个…...

Token原理

Q&#xff1a;分布式场景下如何生成token以及使用token的流程&#xff1a; 在分布式场景下&#xff0c;可以采用以下方式生成 token 和进行权限认证&#xff1a; 1. 生成 token&#xff1a; 使用JWT&#xff08;JSON Web Token&#xff09;生成 token。JWT 是一种基于 JSON …...

③【Java组】蓝桥杯省赛真题 持续更新中...

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ 蓝桥杯真题--持续更新中...一、错误票据题目描…...

linux实验之shell编程基础

这世间&#xff0c;青山灼灼&#xff0c;星光杳杳&#xff0c;秋风渐渐&#xff0c;晚风慢慢 shell编程基础熟悉shell编程的有关机制&#xff0c;如标准流。学习Linux环境变量设置文件及其内容/etc/profile/etc/bashrc/etc/environment~/.profile~/.bashrc熟悉编程有关基础命令…...

C语言小程序:通讯录(静态版)

哈喽各位老铁们&#xff0c;今天给大家带来一期通讯录的静态版本的实现&#xff0c;何为静态版本后面会做解释&#xff0c;话不多说&#xff0c;直接开始&#xff01;关于通讯录&#xff0c;其实也就是类似于我们手机上的通讯录一样&#xff0c;有着各种各样的功能&#xff0c;…...

写CSDN博客两年半的收获--总结篇

&#x1f468;‍&#x1f4bb;作者简介&#xff1a;练习时长两年半的java博主 &#x1f39f;️个人主页&#xff1a;君临๑ ps&#xff1a;点赞是免费的&#xff0c;却可以让写博客的作者开心好几天&#x1f60e; 不知不觉间&#xff0c;在csdn写博客也有两年半的时间了&#x…...

中科亿海微FPGA应用(一、点灯)

1.软件&#xff1a; https://download.csdn.net/download/weixin_41784968/87564071 需要申请license才能使用&#xff1a;软件试用申请_软件试用申请_中科亿海微电子科技&#xff08;苏州&#xff09;有限公司 2.开发板&#xff1a; 芯片EQ6HL45&#xff0c;42.5k LUT。 3…...

ElasticSearch - SpringBoot整合ES:实现搜索结果排序 sort

文章目录00. 数据准备01. Elasticsearch 默认的排序方式是什么&#xff1f;02. Elasticsearch 支持哪些排序方式&#xff1f;03. ElasticSearch 如何指定排序方式&#xff1f;04. ElasticSearch 如何按照相关性排序&#xff1f;05. ElasticSearch 查询结果如何不按照相关性排序…...

IDEA的全新UI可以在配置里启用了,快来试试吧!

刚看到IDEA官方昨天发了这样一条推&#xff1a;IDEA的新UI可以在2022.3版本上直接使用了&#xff01;开启方法如下&#xff1a;打开IDEA的Setting界面&#xff0c;在Appearance & Behavior下有个被标注为Beta标签的New UI菜单&#xff0c;具体如下图&#xff1a;勾选Enable…...

第九章 镜像架构和规划 - 备份处于活动状态时自动进行故障转移

文章目录第九章 镜像架构和规划 - 备份处于活动状态时自动进行故障转移备份处于活动状态时自动进行故障转移备份不活动时的自动故障转移对各种中断场景的镜像响应响应主要中断场景的自动故障转移第九章 镜像架构和规划 - 备份处于活动状态时自动进行故障转移 备份处于活动状态…...

Barra模型因子的构建及应用系列七之Liquidity因子

一、摘要 在前期的Barra模型系列文章中&#xff0c;我们构建了Size因子、Beta因子、Momentum因子、Residual Volatility因子、NonLinear Size因子和Book-to-Price因子&#xff0c;并分别创建了对应的单因子策略&#xff0c;其中Size因子和NonLinear Siz因子具有很强的收益能力…...

走进二叉树的世界 ———性质讲解

二叉树的性质和证明前言1.二叉树的概念和结构特殊的二叉树&#xff1a;二叉树的性质前言 本篇博客主要讲述的是有关二叉树的一些概念&#xff0c;性质以及部分性质的相关证明&#xff0c;如果大伙发现了啥错误&#xff0c;可以在评论区指出&#x1f618;&#x1f618; 1.二叉树…...

【SSM】Spring + SpringMVC +MyBatis 框架整合

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ SSM框架整合一、导入相关依赖二、配置web.xml文…...

【算法基础】一篇文章彻底弄懂Dijkstra算法|多图解+代码详解

博主简介&#xff1a;努力学习的大一在校计算机专业学生&#xff0c;热爱学习和创作。目前在学习和分享&#xff1a;算法、数据结构、Java等相关知识。博主主页&#xff1a; 是瑶瑶子啦所属专栏: 算法 &#xff1b;该专栏专注于蓝桥杯和ACM等算法竞赛&#x1f525;近期目标&…...

第二十三天01MySQL多表查询与事务

目录 1. 多表查询 1.1 概述 1.1.1 数据准备 1.1.2 介绍 1.1.3 分类 1.2 内连接 1.2.1 语法 1.2.2 案例演示 1.3 外连接 1.3.1 语法 1.3.2 案例演示 1.4 子查询 1.4.1 介绍 1.4.2 标量子查询 1.4.3 列子查询 1.4.4 行子查询 1.4.5 表子查询 1.5 案例 1.5.1 介…...

TCP协议详解

1.TCP的准备条件在古代的时候&#xff0c;古人们经常写书信进行交流&#xff0c;写书信的前提是你要知道这份信是要寄给谁在网络中&#xff0c;我们通过ip端口号找对目标对象&#xff0c;但是现在网站一般会对ip端口注册一个域名&#xff0c;所以我们一般就是对域名进行查找&am…...

Activiti7与Spring、Spring Boot整合开发

Activiti整合Spring 一、Activiti与Spring整合开发 1.1 Activiti与Spring整合的配置 1)、在pom.xml文件引入坐标 如下 <properties><slf4j.version>1.6.6</slf4j.version><log4j.version>1.2.12</log4j.version> </properties> <d…...

基于SpringBoot实现冬奥会运动会科普平台【源码+论文】

基于SpringBoot实现冬奥会科普平台演示开发语言&#xff1a;Java 框架&#xff1a;springboot JDK版本&#xff1a;JDK1.8 服务器&#xff1a;tomcat7 数据库&#xff1a;mysql 5.7 数据库工具&#xff1a;Navicat11 开发软件&#xff1a;eclipse/myeclipse/idea Maven包&#…...

一文吃透SpringBoot整合mybatis-plus(保姆式教程)

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏…...

C++ primer plus(第六版)编程练习答案 第4章 复合类型

一、程序清单 arrayone.cpp // arrayone.cpp -- small arrays of integers #include <iostream> int main() {using namespace std;int yams[3]; // creates array with three elementsyams[0] = 7; // assign value to first elementyams[1] = 8;yams[2] = 6;i…...

Kafka源码分析之Producer(一)

总览 根据kafka的3.1.0的源码example模块进行分析&#xff0c;如下图所示&#xff0c;一般实例代码就是我们分析源码的入口。 可以将produce的发送主要流程概述如下&#xff1a; 拦截器对发送的消息拦截处理&#xff1b; 获取元数据信息&#xff1b; 序列化处理&#xff1b;…...

springboot校友社交系统

050-springboot校友社交系统演示录像开发语言&#xff1a;Java 框架&#xff1a;springboot JDK版本&#xff1a;JDK1.8 服务器&#xff1a;tomcat7 数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09; 数据库工具&#xff1a;Navicat11 开发软件&#xff1a;e…...

python flask项目部署

flask上传服务器pyhon安装下载Anacondasudo wget https://mirrors.tuna.tsinghua.edu.cn/anaconda/archive/Anaconda3-5.3.1-Linux-x86_64.sh可根据需要安装对应的版本https://repo.anaconda.com/archive/解压anaconda压缩包bash Anaconda3-5.3.1-Linux-x86_64.sh解压过程中会…...

常见排序算法(C语言实现)

文章目录排序介绍插入排序直接插入排序希尔排序选择排序选择排序堆排序交换排序冒泡排序快速排序递归实现Hoare版本挖坑法前后指针版本非递归实现Hoare版本挖坑法前后指针版本快排的优化三值取中小区间优化归并排序递归实现非递归实现计数排序排序算法复杂度及稳定性分析不同算…...

基于jsp+ssm+springboot的小区物业管理系统【设计+论文+源码】

摘 要随着科学技术的飞速发展&#xff0c;各行各业都在努力与现代先进技术接轨&#xff0c;通过科技手段提高自身的优势&#xff1b;对于小区物业管理系统当然也不能排除在外&#xff0c;随着网络技术的不断成熟&#xff0c;带动了小区物业管理系统&#xff0c;它彻底改变了过去…...

Elasticsearch 学习+SpringBoot实战教程(三)

需要学习基础的可参照这两文章 Elasticsearch 学习SpringBoot实战教程&#xff08;一&#xff09; Elasticsearch 学习SpringBoot实战教程&#xff08;一&#xff09;_桂亭亭的博客-CSDN博客 Elasticsearch 学习SpringBoot实战教程&#xff08;二&#xff09; Elasticsearch …...

try-with-resource

try-with-resource是Java 7中引入的新特性&#xff0c;它可以方便地管理资源&#xff0c;自动关闭资源&#xff0c;从而避免了资源泄漏的问题。 作用 使用try-with-resource语句可以简化代码&#xff0c;避免了手动关闭资源的繁琐操作&#xff0c;同时还可以保证资源的正确关闭…...

安徽平台网站建设公司/重庆seo团队

JSONP 和 JSON 的关系 事实上&#xff0c;JSONP 和 JSON 没有关系。若强行要有关系&#xff0c;也只能说 JSONP 这个技术使用了 JSON 这种数据格式。JSON 是一种数据交换格式。JSONP 是一种非官方跨域数据交互协议&#xff0c;是一种技术。 跨域是什么&#xff1f; 上面说了 JS…...

山西今日头条新闻消息/网站关键词优化排名怎么做

作为原始的调试方式&#xff0c;开发者可以使用帮助函数中的DBGPRINTF()在应用中的任何位置将调试信息打印到模拟器的输出窗口中。模拟器的输出窗口还会显示大量的AEE层运行时产生的信息&#xff0c;这些信息的含义如表16-1所示。 标识 含义 #*gBI BREW 进行初始化 #*gEX 退出B…...

洛阳网站建设多少钱/app推广平台排行榜

最最简单的数据结构&#xff0c;数据结构入门必备&#xff0c;新手必备顺序表! 目录本节目标1.线性表概念顺序表实现静态顺序表动态顺序表接口实现本节目标 了解线性表结构能够自己实现顺序表顺序表oj题 1.线性表概念 1线性表线性表&#xff08;linear list&#xff09; 是…...

站酷官网/seo设置是什么

前言 一次偶然&#xff0c;从朋友那里得到一份“java高分面试指南”&#xff0c;里面涵盖了25个分类的面试题以及详细的解析&#xff1a;JavaOOP、Java集合/泛型、Java中的IO与NIO、Java反射、Java序列化、Java注解、多线程&并发、JVM、Mysql、Redis、Memcached、MongoDB、…...

怎么做动态网站页面/百度上做优化

类似py2exe软件真的能保护python源码吗 背景 最近写了个工具用于对项目中C/C文件的字符串常量进行自动化加密处理&#xff0c;用python写的&#xff0c;工具效果不错&#xff0c;所以打算在公司内部推广。为了防止代码泄露就考虑不采用直接给源码方式&#xff0c;而python二进制…...

网站运营与规划/北京网站seo设计

每个月总有那么几天&#xff0c;状态不好。不想干活&#xff0c;在电脑前刷刷天涯&#xff0c;看看NBA数据&#xff0c;想把一天的时间都打发掉。周末又去看了一下房子&#xff0c;我只想说CTMDZF,一群杀人不眨眼的恶魔&#xff0c;心安理得得奴役“贫民”. 反正现在是已经…...