详细解释一下Spring是如何解决循环依赖问题的
Spring是如何解决循环依赖问题的?
我们都知道,如果在代码中,将两个或多个Bean互相之间持有对方的引用就会发生循环依赖。循环的依赖将会导致注入死循环,这是Spring发生循环依赖的原因
循环依赖有三种形态
1.互相依赖:
A依赖B , B依赖A , 他们之间形成了循环依赖

2.间接依赖:
A依赖B ,B依赖C , C又依赖A,形成了循环依赖

3.自我依赖:
A依赖A形成了循环依赖

Spring中的三级缓存
Spring中设计了三级缓存来解决循环依赖问题,当我们去调用getBean()方法的时候:
1> Spring会先从一级缓存中去找到目标Bean,如果发现一级缓存中没有便会去二级缓存中去找
2> 如果一、二级缓存中都没有找到,意味着该目标Bean还没有实例化(早期Bean)。于是,Spring容器会实例化目标Bean,然后将目标Bean放入到二级缓存中,同时,加上标记是否存在循环依赖。如果不存在循环依赖便会将目标Bean存入到二级缓存
3> 否则,便会标记该Bean存在循环依赖,然后将等待下一次轮询赋值,也就是解析@Autowired注解。等@Autowird(成熟Bean),会将目标Bean存入到一级缓存
第三级缓存的作用是用来存储代理Bean的,当调用getBean()方法之后,发现目标Bean需要通过代理工程来创建,此时会将创建好的实例保存到三级缓存,最终也会赋值好的Bean同步到一级缓存中
早期Bean:刚初始化的Bean
成熟Bean:完成注入的Bean
在Spring三级缓存的设计下,我画了一张图来描述一下其工作原理:

Spring本身只能解决单实例存在的循环依赖问题,存在以下四种情况需要人为干预:
1> 多实例的Setter注入导致的循环依赖,需要把Bean改成单例
2> 构造器注入导致的循环依赖,可以通过@Lazy注解
3> DependsOn导致的循环依赖,找到注解循环依赖的地方,迫使它不循环依赖
4> 单例的代理对象Setter注入导致的循环依赖
① 可以使用Lazy注解
② 或者使用@DependsOn注解指定加载先后关系
在实际开发中,出现循环依赖的根本原因还是在代码设计的时候,因为模块的耦合度较高,依赖关系复杂导致的,我们应该尽可能地从系统设计角度去考虑模块之间的依赖关系,避免循环依赖地问题。
三级缓存的核心思想:就是把Bean的实例化和Bean里面的依赖注入进行分离
总结:
① 采用一级缓存存储成熟Bean实例,采用二级缓存来存储早期Bean实例
② 通过早期Bean实例作为突破口,解决循环依赖问题。
③ 至于第三级缓存,主要是解决代理对象的循环依赖问题。
详细解释一下Spring是如何解决循环依赖问题的的更多相关文章
- Spring是如何解决循环依赖的
前言 在面试的时候这两年有一个非常高频的关于spring的问题,那就是spring是如何解决循环依赖的.这个问题听着就是轻描淡写的一句话,其实考察的内容还是非常多的,主要还是考察的应聘者有没有研究过s ...
- spring: 我是如何解决循环依赖的?
1.由同事抛的一个问题开始 最近项目组的一个同事遇到了一个问题,问我的意见,一下子引起的我的兴趣,因为这个问题我也是第一次遇到.平时自认为对spring循环依赖问题还是比较了解的,直到遇到这个和后面的 ...
- Spring 是如何解决循环依赖的?
前言 相信很多小伙伴在工作中都会遇到循环依赖,不过大多数它是这样显示的: 还会提示这么一句: Requested bean is currently in creation: Is there an ...
- 听说你还不知道Spring是如何解决循环依赖问题的?
Spring如何解决的循环依赖,是近两年流行起来的一道Java面试题. 其实笔者本人对这类框架源码题还是持一定的怀疑态度的. 如果笔者作为面试官,可能会问一些诸如"如果注入的属性为null, ...
- Spring三级缓存解决循环依赖
前提知识 1.解决循环依赖的核心依据:实例化和初始化步骤是分开执行的 2.实现方式:三级缓存 3.lambda表达式的延迟执行特性 spring源码执行逻辑 核心方法refresh(), popula ...
- Spring 如何解决循环依赖问题?
在关于Spring的面试中,我们经常会被问到一个问题,就是Spring是如何解决循环依赖的问题的. 这个问题算是关于Spring的一个高频面试题,因为如果不刻意研读,相信即使读过源码,面试者也不一定能 ...
- Spring解决循环依赖,你真的懂了吗?
导读 前几天发表的文章SpringBoot多数据源动态切换和SpringBoot整合多数据源的巨坑中,提到了一个坑就是动态数据源添加@Primary接口就会造成循环依赖异常,如下图: 这个就是典型的构 ...
- Spring如何解决循环依赖,你真的懂了?
导读 前几天发表的文章SpringBoot多数据源动态切换和SpringBoot整合多数据源的巨坑中,提到了一个坑就是动态数据源添加@Primary接口就会造成循环依赖异常,如下图: 这个就是典型的构 ...
- 浅谈Spring解决循环依赖的三种方式
引言:循环依赖就是N个类中循环嵌套引用,如果在日常开发中我们用new 对象的方式发生这种循环依赖的话程序会在运行时一直循环调用,直至内存溢出报错.下面说一下Spring是如果解决循环依赖的. 第一种: ...
- Spring ioc(4)---如何解决循环依赖
前面说到对象的创建,那么在创建的过程中Spring是怎么又是如何解决循环依赖的呢.前面提到有个三级缓存.就是利用这个来解决循环依赖.打个比方说实例化A的时候,先将A创建(早期对象)放入一个池子中.这个 ...
随机推荐
- Netty实战(五)
一.什么是ByteBuf 我们前面说过,网络数据的基本单位总是字节.Java NIO 提供了 ByteBuffer 作为它的字节容器,但是这个类使用起来过于复杂,而且也有些繁琐.ByteBuffer ...
- UpSetR:多数据集绘图可视化处理利器
说到集合数据可视化,我们第一时间想到的就是韦恩图.在 NGS 相关的研究中,韦恩图用来直观表征不同的集合之间元素重叠关系,是经常在文献中出现的图. 在集合数少的时候韦恩图是很好用的,但是当集合数多比如 ...
- error: #20: identifier "arm_cfft_instance_f64" is undefined
在使用Keil5的过程中,偶尔遇到这个问题,以及类似的问题,报错的数量大概200多个. error type>(42): error: #20: identifier "arm_cff ...
- Python进行大文件的备份
Python进行大文件的备份的思路:每次仅从原文件中读取指定字符的内容后写入新文件,然后循环操作. def copy_big_file(): # 接收用户输入的文件名 old_file = input ...
- 推荐一个 C#写的 支持OCR的免费通用扫描仪软件
NAPS2是一个开源免费软件,体积只有6M不到,支持运行在 Windows, Mac 和 Linux操作系统中,默认就带有简体中文界面,官方默认就提供绿色版,所以解压即可使用,直接可以从官方网站下载: ...
- 离线安装rpm包以及自建yum仓库
离线安装rpm包以及自建yum仓库 离线安装rpm yum支持如下参数 --downloadnoly 只下载不安装 --downloaddir=directory 下载到指定目录下 因此可以在线下载好 ...
- 给你的 Discord 接入一个既能联网又能画画的 ChatGPT
如果有这样一款 Discord 机器人,它既能访问互联网,又能绘画,还能给 YouTube 视频提供摘要.最重要的是,它是完全免费的,不需要提供 OpenAI 的 API Key,我就问你香不香? 现 ...
- 【AGC】云监控日志服务查询不到Logger日志相关问题2
[关键字] AGC.云监控.日志服务 [问题描述] 之前有开发者反馈在使用AGC云监控,填写了Logger日志,但是在云监控的日志服务查不到的问题.具体如下所述: 云函数按要求写了Logger日志, ...
- 如何通过AWS的AmazonSageMaker进行机器学习
目录 <如何通过 AWS 的 Amazon SageMaker 进行机器学习> 一.引言 随着人工智能和机器学习的发展,越来越多的企业和机构开始使用这些技术来进行各种应用场景的处理和分析. ...
- PostgreSQL 性能优化: EXPLAIN 使用教程
使用 EXPLAIN EXPLAIN基础 代价估计 启动开销 总开销 计划结点输出行数 计划结点输出行宽 执行统计 实际启动开销 实际总开销 实际输出行数 实际执行次数 I/O统计 共享块命中数 共享 ...