Spring为什么要用三级缓存解决循环依赖?

1.什么是循环依赖

本文为了方便说明,先设置两个业务层对象,命名为AService和BService。其中Spring是如何把一个Bean对象创建出来的,其生命周期如下:

构造方法–> 不同对象 --> 注入依赖 -->初始化前 --> 初始化后–>放入单例池Map(一级缓存)—>Bean对象

Map的数据结构为,key表示单例的名称,而value是一个对象。其中单例池就是所谓的一级缓存。

循环依赖,如AService中依赖了一个BService,BService也依赖了AService,AService和BService相互依赖。这里换出现一个问题,但是Spring使用三级缓存给解决了。
首先我们要看的是为什么出现循环?
首先在AService中引入了@Componet注解,那AService的生命周期会交给Spring管理。AService生命周期如下,本文把该例子定义为示例1。

AService生命周期

1.实例化 --->AService的普通对象
2.填充BService(添加了AutoWired)-->从单例池中获取BService(此时单例池中可能还没存放BService的Bean)-->创建BService
  当单例池中可能还没存放BService的Bean时,触发创建BService生命周期,步骤和AService一致。如下所示:
  2.1.实例化--->BService的普通对象
  2.2.填充AService(添加了AutoWired)-->单例池(此时AAService也在创建中,还没放入单例池,如果创建的话,就会出现循环依赖)
  2.3.填充其他属性
  2.4.其他步骤(包括AOP)
  2.5.加入到单例池中
3.填充其他属性
4.其他步骤(包括AOP)
5.加入到单例池中	

在这里插入图片描述

简单解决示例1中的问题

主要是定义个Map存放第一步生成的普通对象。Map的名称暂且为boziMap<beanName,普通对象>,示例2如下所示:

AService生命周期

1.实例化 --->AService的普通对象---->存入boziMap<beanName,AService的普通对象>
2.填充BService(添加了AutoWired)-->从单例池中获取BService(此时单例池中可能还没存放BService的Bean)-->创建BService
  当单例池中可能还没存放BService的Bean时,触发创建BService生命周期,步骤和AService一致。如下所示:
  2.1.实例化--->BService的普通对象
  2.2.填充AService(添加了AutoWired)-->单例池(此时AAService也在创建中,还没放入单例池,如果创建的话,就会出现循环依赖)--->boziMap中取AService对象。
  2.3.填充其他属性
  2.4.其他步骤(包括AOP)
  2.5.加入到单例池中
3.填充其他属性
4.其他步骤(包括AOP)--> AService代理对象(AServiceProxy)
5.加入到单例池中	

其中AService对象的代理对象和AService的代理对象如下所示:
在这里插入图片描述

示例2中通过一个boziMap打破了循环创建bean对象,而产生的循环依赖。而在AOP过程中,步骤2.2.填充AService时,应该是把4.其他步骤,AService代理对象赋值给步骤2.2.填充AService的普通对象。所以要对示例2进行优化,把AOP放到第二步,先判断是否出现循环依赖,在进行AOP,再把AService的代理对象放入biziMap中。得到示例3:

AService生命周期
0.creatingSet[AService]表示正在创建中的Bean。
1.实例化 --->AService的普通对象
2.填充BService(添加了AutoWired)-->从单例池中获取BService
	(此时单例池中可能还没存放BService的Bean)-->创建BService
  当单例池中可能还没存放BService的Bean时,触发创建BService生命周期,步骤和AService一致。如下所示:
  2.1.实例化--->BService的普通对象
  2.2.填充aService(添加了AutoWired)-->单例池-->creatingSet中是否存在AService-->AOP-->AService的代理对象-->最后赋值给BService中的属性aService
  2.3.填充其他属性
  2.4.其他步骤
  2.5.加入到单例池中
3.填充其他属性
4.其他步骤(包括AOP)--> AService代理对象(AServiceProxy)
5.加入到单例池中	

此时加大难度,多加一个CService,C中依赖的A,而A中也依赖了C,根据示例3,会得到如下步骤。把本例子定义为示例4,该例子中,填充bService和cService会反复创建代理对象。

AService生命周期
0.creatingSet[AService]表示正在创建中的Bean。
1.实例化 --->AService的普通对象
2.填充BService,CService(添加了AutoWired)-->从单例池中获取BService
	(此时单例池中可能还没存放BService的Bean)-->创建BService
  当单例池中可能还没存放BService的Bean时,触发创建BService生命周期,步骤和AService一致。如下所示:
  2.1.实例化--->BService的普通对象
  2.2.填充aService(添加了AutoWired)-->单例池-->creatingSet中是否存在AService-->AOP-->AService的代理对象-->最后赋值给BService中的属性aService
  2.3.填充其他属性
  2.4.其他步骤
  2.5.加入到单例池中
  填充CService,CService的生命周期
  2.1.实例化-->BService的普通对象
  2.2.填充aService(添加了AutoWired)-->creatingSet中是否存在AService-->AOP-->AService的代理对象-->最后赋值给CService中的属性aService
  2.3.填充其他属性
  2.4.其他步骤
  2.5.加入到单例池中
3.填充其他属性
4.其他步骤(包括AOP)--> AService代理对象(AServiceProxy)
5.加入到单例池中	

为了解决这个问题,使用二级缓存存放A的代理对象。

AService生命周期
0.creatingSet[AService]表示正在创建中的Bean。
1.实例化 --->AService的普通对象
2.填充BService,CService(添加了AutoWired)-->从单例池中获取BService
	(此时单例池中可能还没存放BService的Bean)-->创建BService
  当单例池中可能还没存放BService的Bean时,触发创建BService生命周期,步骤和AService一致。如下所示:
  2.1.实例化--->BService的普通对象
  2.2.填充aService(添加了AutoWired)-->单例池-->creatingSet中是否存在AService-->循环依赖
  --->earlySingletonObjects是否存在aService的代理对象,存在返回bean即可
  -->AOP-->AService的代理对象-->存入二级缓存,earlySingletonObjects<beanName,AService的代理对象>-->最后赋值给BService中的属性aService
  2.3.填充其他属性
  2.4.其他步骤
  2.5.加入到单例池中
  填充CService,CService的生命周期
  2.1.实例化-->BService的普通对象
  2.2.填充aService(添加了AutoWired)-->单例池-->creatingSet中是否存在AService-->earlySingletonObjects
  2.3.填充其他属性
  2.4.其他步骤
  2.5.加入到单例池中
3.填充其他属性
4.其他步骤(包括AOP)--> AService代理对象(AServiceProxy)
5.加入到单例池中	

三级缓存

三级缓存为了打破循环,在第一步骤中生成的不同对象,用三级缓存保存起来。三级缓存在Spring‘源码中可singletonFactories,是一个Map,其中它的value存的是一个lambda表达式,其中的逻辑是,判断是否需要AOP,需要则返回一个代理对象,反之则返回Service的普通对象。’
在这里插入图片描述

小结

一级缓存的作用就是保存经过完整生命周期的Bean对象。
二级缓存,早期由于出现循环依赖,保存那些还没完整走完bean生命周期的bean对象,是为了给其他Bean填充属性时使用的代理对象赋值。
三级缓存,在出现循环依赖时,如果二级缓存中没有存有对应的bean对象,需要通过三级缓存去判断,是否需要AOP,是则需要返回代理对象,否则需要返回普通对象。三级返回的的结果最终还是存在二级缓存中。Spring的核心源码实现如下所示。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/949028.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

R shiny app | 网页应用 空格分隔的文本文件在线转csv

shiny 能快速把R程序以web app的形式提供出来&#xff0c;方便使用&#xff0c;降低技术使用门槛。 本文提供的示例&#xff1a;把空格分隔的txt文件转为逗号分隔的csv文件。 前置依赖&#xff1a;需要有R环境(v4.2.0)&#xff0c;安装shiny包(v1.9.1)。括号内是我使用的版本…

LLM - 使用 LLaMA-Factory 部署大模型 HTTP 多模态服务 教程 (4)

欢迎关注我的CSDN:https://spike.blog.csdn.net/ 本文地址:https://spike.blog.csdn.net/article/details/144881432 大模型的 HTTP 服务,通过网络接口,提供 AI 模型功能的服务,允许通过发送 HTTP 请求,交互大模型,通常基于云计算架构,无需在本地部署复杂的模型和硬件,…

【《python爬虫入门教程11--重剑无峰168》】

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 【《python爬虫入门教程11--selenium的安装与使用》】 前言selenium就是一个可以实现python自动化的模块 一、Chrome的版本查找&#xff1f;-- 如果用edge也是类似的1.chrome…

自定义Bitmap

使用场景 Bitmap 是一种使用 位数组&#xff08;bit array&#xff09; 来表示数据的结构&#xff0c;每一位&#xff08;bit&#xff09;表示一个值的状态。由于每个位只占用 1 bit&#xff0c;因此Bitmap 的存储效率非常高&#xff0c;特别适用于大数据去重、标记布尔值状态…

公司资产网站

本文结尾处获取源码。 本文结尾处获取源码。 本文结尾处获取源码。 一、相关技术 后端&#xff1a;Java、JavaWeb / Springboot。前端&#xff1a;Vue、HTML / CSS / Javascript 等。数据库&#xff1a;MySQL 二、相关软件&#xff08;列出的软件其一均可运行&#xff09; I…

第431场周赛:最长乘积等价子数组、计算字符串的镜像分数、收集连续 K 个袋子可以获得的最多硬币数量、不重叠区间的最大得分

Q1、最长乘积等价子数组 1、题目描述 给你一个由 正整数 组成的数组 nums。 如果一个数组 arr 满足 prod(arr) lcm(arr) * gcd(arr)&#xff0c;则称其为 乘积等价数组 &#xff0c;其中&#xff1a; prod(arr) 表示 arr 中所有元素的乘积。gcd(arr) 表示 arr 中所有元素的…

掌握RabbitMQ:全面知识点汇总与实践指南

前言 RabbitMQ 是基于 AMQP 高级消息队列协议的消息队列技术。 特点&#xff1a;它通过发布/订阅模型&#xff0c;实现了服务间的高度解耦。因为消费者不需要确保提供者的存在。 作用&#xff1a;服务间异步通信&#xff1b;顺序消费&#xff1b;定时任务&#xff1b;请求削…

国内Ubuntu环境Docker部署Stable Diffusion入坑记录

国内Ubuntu环境Docker部署Stable Diffusion入坑记录 本文旨在记录使用dockerpython进行部署 stable-diffusion-webui 项目时遇到的一些问题&#xff0c;以及解决方案&#xff0c;原项目地址: https://github.com/AUTOMATIC1111/stable-diffusion-webui 问题一览&#xff1a; …

SpringBoot3-深入理解自动配置类的原理(尚硅谷SpringBoot3-雷神)

文章目录 目录了解自动配置 一、导入对应场景的Mean依赖&#xff1a;1、引入依赖**找到自动配置类的所有配置都存放在哪里** 二、编写主程序&#xff1a;SpringBootApplication观察源码时所需要知道的几个核心注解&#xff1a;1、观察SpringBootApplication源码都做了什么 三、…

【沉默的羔羊心理学】汉尼拔的“移情”游戏:操纵与理解的艺术,精神分析学视角下的角色互动

终极解读《沉默的羔羊》&#xff1a;弗洛伊德精神分析学视角下的深层剖析 关键词 沉默的羔羊弗洛伊德精神分析学角色心理意识与潜意识性别与身份 弗洛伊德精神分析学简介 弗洛伊德的精神分析学是心理学的一个重要分支&#xff0c;主要关注人类行为背后的无意识动机和冲突。…

字玩FontPlayer开发笔记3 性能优化 大量canvas渲染卡顿问题

字玩FontPlayer开发笔记3 性能优化 大量canvas渲染卡顿问题 字玩FontPlayer是笔者开源的一款字体设计工具&#xff0c;使用Vue3 ElementUI开发&#xff0c;源代码&#xff1a; github: https://github.com/HiToysMaker/fontplayer gitee: https://gitee.com/toysmaker/fontpl…

javaEE-网络编程-3 UDP

目录 Socaket套接字 UDP数据报套字节编程 1.DatagrameSocket类 DatagramSocaket构造方法: DatagramSocaket常用方法&#xff1a; 2.DatagramPacket类 DatagramPacket构造方法&#xff1a; UDP回显服务器实现 UDP服务端实现&#xff1a; 创建一个Socket类对象&#xf…

Linux:操作系统不朽的传说

操作系统是计算机的灵魂&#xff0c;它掌控着计算机的硬件和软件资源&#xff0c;为用户和应用程序提供了一个稳定、高效、安全的运行环境。 在众多操作系统中&#xff0c;Linux 的地位举足轻重。它被广泛应用于服务器、云计算、物联网、嵌入式设备等领域。Linux 的成功离不开…

模拟出一个三维表面生成表面点,计算体积,并处理边界点

python代码 生成表面点,计算体积,并处理边界点,最终模拟出一个三维表面。 步骤: 初始参数设置: initial_fixed_point:一个初始固定点的坐标。 slop_thre:坡度阈值。 v_thre:体积阈值。 slope_rad:将坡度从度转换为弧度。 step_size:步长。 lam_x, lam_y:泊松分布的…

STM32拓展 低功耗案例1:睡眠模式 (register)

需求描述 让MCU进入睡眠模式&#xff0c;然后通过串口发送消息来唤醒MCU退出睡眠模式。观察LED在进入休眠模式后是否仍然开启。 思考 首先睡眠模式&#xff0c;唤醒的条件是中断&#xff0c;外部内部都可以&#xff0c;这里的串口接收中断时内部中断。 拓展&#xff1a;中断…

vue 基础参数增加多语言配置

js 对数组的增删改查 字段在数据库存储为nvarchar &#xff0c;varchar存储波斯语会乱码 数组格式&#xff1a; {"en": [{"type": "10","value": "Confirm","color": ""},{"type": "…

[桌面运维]windows自动设置浅深色主题

设置自动浅色/深色主题 我看很多up主的教程过于繁琐&#xff0c;需要添加四个功能&#xff0c;并且有些还不能生效&#xff01; 大多数都是教程&#xff1a; 自动任务栏浅色 add HKCUSOFTWAREMicrosoftWindowsCurrentVersionThemesPersonalize/v SystemUsesLightTheme /t …

[ubuntu-22.04]ubuntu不识别rtl8153 usb转网口

问题描述 ubuntu22.04插入rtl8153 usb转网口不识别 解决方案 安装依赖包 sudo apt-get install libelf-dev build-essential linux-headers-uname -r sudo apt-get install gcc-12 下载源码 Realtek USB FE / GBE / 2.5G / 5G Ethernet Family Controller Softwarehttps:/…

基于Python的考研学习系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏&#xff1a;…

图漾相机基础操作

1.客户端概述 1.1 简介 PercipioViewer是图漾基于Percipio Camport SDK开发的一款看图软件&#xff0c;可实时预览相机输出的深度图、彩色图、IR红外图和点云图,并保存对应数据&#xff0c;还支持查看设备基础信息&#xff0c;在线修改gain、曝光等各种调节相机成像的参数功能…