首页 > 其他分享 >Spring循环依赖

Spring循环依赖

时间:2023-02-06 21:11:47浏览次数:44  
标签:缓存 依赖 Spring 循环 Bean getSingleton 三级

Spring循环依赖面试中也会被常常问到。但是它的整个过程很多人都不知道,什么叫循环依赖呢。多个Bean之间相互依赖,形成一个闭环。如下图(A,B,C分别为Spring容器中3个Bean)就能很好的描述。(PS必须保证默认的Bean都是单例的循环依赖才成立)。

 

上面是对Spring循环依赖的简单解释,下图是Spring官网说明。

 

上图中重点翻译就是:循环依赖不支持构造方法的注入,使用构造方法注入会抛出BeanCurrentlyInCreationException异常 。Spring不推荐构造注入,只支持setter注入。

Spring是怎么支持和解决循环依赖的呢?

 

它目前解决的方案是靠自身容器中的3个Map来解决的,也称为三级缓存,如下图:

 

第一级缓存(也叫单例池)singleObjects:存放已经经历完整生命周期的Bean对象。

第二级缓存:earlySingletonObjects,存放早期暴漏的Bean对象,Bean周期未结束(属性未完全填充)。

第三级缓存:singletonFactories,存放可以生成Bean的工厂。

那么下面我们就通过源码来说明Spring是如何通过上面三级缓存来解决循环依赖的,我们就用2个对象来演示和理解循环依赖。

1:A创建过程中需要B,于是A将自己放到三级缓存里面,去实例化B。

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

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

源码分析主要关注如下方法的流程:

 

 

 

 

1:调用doCreateBean()方法,想要获取a ,于是调用getSingleton()方法从缓存中获取a 。

 

2:getSingleton()方法从一级缓存中找,没有找到返回null。

3:如果getSingleton()方法中获取到的a为null ,于是走对应的逻辑处理,调用getSingleton的重载的方法(参数为ObjectFactory)。

 

4:在getSingleton()中将a添加到一个集合中,用于标记a正在创建中,然后调用匿名内部类的creatBean方法。

5:进入AbstractAutowireCapableBeanFactory的doCreateBean,创建出a的实例然后将a添加到三级缓存中。

 

6:对a进行属性的填充,此时检测到a依赖于b,于是开始找b 。

 

 

7:找b的时候调用doGetBean()方法和上面创建a的过程一样,到缓存中找b,没有则创建,然后给b填充属性。

 

8:此时b依赖于a,调用getSingleton获取a,依次从一级,二级,三级缓存中找,此时从三级缓存中获取a的创建工厂,通过创建工厂获取到singeltonObject,此时这个singletonObject指向的就是上面的doCreateBean方法中获取的实例化的a。

 

 

9:这样b就获取到了a的依赖,于是b顺利完成了实例化,并将a从三级缓存移到二级缓存中。

 

10:随后a继续属性填充的工作,此时也获取了b,然后a也就完成了创建,回到getSingleton方法中继续执行,将a从二级缓存移动到一级缓存。

 

具体为什么用三级缓存不用二级缓存去解决,后面章节分析,欢迎转发,评论,点赞

微信公众号搜索:程序员xiaozhang 。如果遇到Spring的问题也可以私信我 能帮忙解决的尽量解决。

 

 

标签:缓存,依赖,Spring,循环,Bean,getSingleton,三级
From: https://www.cnblogs.com/scott1102/p/17096114.html

相关文章

  • SpringBoot2.5.6集成mybatis
    1.应用依赖<dependency><groupId>org.mybatis.spring.boot</groupId><artifactId>mybatis-spring-boot-starter</artifactId><version>2.2.0</version......
  • Spring在Filter中记录Web请求Request和返回Response的内容及时长
    1简介在SpringMVC中,我们有时需要记录一下请求和返回的内容,方便出现问题时排查。比较Header、RequestBody等。这些在Controller也可以记录,但在Filter中会更方便。而我们......
  • SpringBoot响应Json数据乱码通过配置解决
    场景实现把SpringBoot的response编码设置为utf-8找到application.properties配置文件添加如下:#设置响应为utf-8spring.http.encoding.force-response=true 再次刷新浏览器......
  • SpringBoot中自定义消息转化器
    场景1.SpringBoot自动配置了消息转化器。2.自定义消息转化器,只需要在类中添加消息转化器的@Bean,就会被SpringBoot自动加入到容器中。实现新建Controllerpackagecom.exampl......
  • SpringBoot+MyBatis的动态SQL、使用动态SQL时List传值错误解决方案
    目录实现动态SQL的四种方式:1、XML配置2、脚本SQL3、在方法中构建SQL4、结构化SQL关于动态SQL的List传值错误问题1、错误代码2、解决错误实现动态SQL的四种方式:1、XML配置......
  • Spring Cloud Alibaba 在 Proxyless Mesh 上的探索
    站在2023年的今天,ServiceMesh早已不是一个新兴的概念,回顾过去6年多的发展历程,ServiceMesh从一经推出就受到来自全世界的主流技术公司关注和追捧。2016年作为S......
  • spring 重复注解和aop拦截的实现示例
    前言:1:jdk1.8开始支持重复注解@Repeatable实现2:aop拦截需要拦截当前注解和@Repeatable指向的包装注解才可以完全拦截到,因为:1.当在在方法上只有一个注解时,aop拦截......
  • 详解Spring AOP自定义可重复注解没有生效问题
    目录1.问题背景2.不啰嗦,上代码3.问题排查3.1是不是切点写得有问题,于是换成如下形式:3.2是不是使用的地方不是代理对象4.问题原因 1.问题背景工作中遇......
  • Spring AOP使用接口方式实现
    目录一.环境准备二、spring接口方式实现aop步骤1.业务接口实现2.业务类3.通知类4.自定义切##点5.配置xml文件6.方法入口三.分析Spring提供了很多的......
  • 循环链表的创建、插入、删除、逆序、显示(C++实现)
    对于单链表,因为每一个结点仅仅存储了向后的指针。到了尾标志就停止了向后链的操作,这样,其中某一结点就无法找到它的前驱结点了。对于单链表的操作大家能够看我的这篇博客htt......