Spring-bean的循环依赖

news2025/1/16 5:07:34

文章目录

    • 什么是Spring的循环依赖?
    • Setter 方法循环依赖
      • 三级缓存
      • Spring 是如何利用三级缓存解决 Bean 的循环依赖?
      • 有了三级缓存为什么还需要二级缓存?
    • 构造器循环依赖
    • 总结

什么是Spring的循环依赖?

在 Spring 框架中,循环依赖是指两个或多个 Bean 相互依赖,形成一个闭环。

  • Setter 方法循环依赖:Bean A 通过 setter 方法注入 Bean B,Bean B 又通过 setter 方法注入 Bean A。(即@Autowired)
  • 构造器循环依赖:Bean A 需要 Bean B 的实例,而 Bean B 又需要 Bean A 的实例。

Setter 方法循环依赖

在这里插入图片描述

三级缓存

三级缓存是 Spring 处理 Setter 循环依赖的关键机制

  • Singleton Objects(一级缓存):存放已完全初始化的单例 Bean。
  • Early Singleton Objects(二级缓存):缓存早期的 Bean 对象(就是这些 Bean 已经被实例化,但是还没有被完全初始化)。也可避免重复通过工厂获取早期引用。
  • Singleton Factories(三级缓存):存储可以创建单例Bean实例的工厂,可以创建代理对象(二级缓存不能解决的),普通对象

Spring 是如何利用三级缓存解决 Bean 的循环依赖?

三级缓存是 Spring 框架中用于解决 Bean 循环依赖的一种机制。

一级缓存
在这里插入图片描述

二级缓存:
在这里插入图片描述
三级缓存

二级缓存用于存储已经创建但未完全初始化的早期 Bean 引用。但是在 Bean 被完全初始化之前,早期引用可能需要通过代理对象(如 AOP 代理)创建。三级缓存正是用于存储这种生成早期引用的工厂方法。

通过三级缓存,Spring 可以在 ObjectFactory 中定义如何创建早期引用(如处理代理),这样就可以灵活地在需要时(特别是在循环依赖情况下)提供一个代理后的早期引用给其他 Bean 使用。
在这里插入图片描述

有了三级缓存为什么还需要二级缓存?

二级缓存的引入可以防止多次使用对象工厂(ObjectFactory)创建多个对象实例,从而避免不必要的多例问题

首次访问早期引用: 当一个 Bean 正在初始化过程中(比如 A 依赖 B ,但 B 尚未完全初始化)时,Spring 会首先检查一级缓存(完全初始化的单例对象)中是否有 B。如果没有,再检查二级缓存(早期暴露的对象引用)。如果二级缓存中也没有,再通过三级缓存的 ObjectFactory 获取 B 的早期引用。 一旦通过 ObjectFactory 获取了 B 的早期引用,Spring 会将这个引用放入二级缓存。这样,后续对 B 的早期引用访问就不会再触发 ObjectFactory,而是直接从二级缓存中获取。

构造器循环依赖

三级缓存机制可以自动处理基于单例作用域的 setter 方法循环依赖,但无法解决由构造器注入引起的循环依赖。

比如:

@Component
public class A {
    private B b;

    @Autowired
    public A(B b) {
        this.b = b;
    }
}

@Component
public class B {
    private A a;

    @Autowired
    public B(A a) {
        this.a = a;
    }
}

解决方法:使用 @Lazy 注解,在构造器注入中使用 @Lazy 注解,可以推迟依赖的初始化,从而打破循环。@Lazy 注解表示在实际使用时才创建实例,而不是在启动时。

@Component
public class A {
    private B b;

    @Autowired
    // 添加注解 @Lazy
    public A(@Lazy B b) {
        this.b = b;
    }
}

@Component
public class B {
    private A a;

    @Autowired
    public B(@Lazy A a) {
        this.a = a;
    }
}

总结

  • Setter 注入循环依赖:Spring 通过三级缓存机制自动处理

  • 构造器注入循环依赖:Spring 无法自动解决,使用 @Lazy 注解


❤觉得有用的可以留个关注丫~❤

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

萌化人心的AI宠物视频:1条爆款35万赞,带你轻松涨粉10万(附教程)

萌宠系的小动物配上可爱的 BGM,场景情绪感直接拉满。用 AI 做超萌宠物视频,涨粉确实香, 分享一个有趣的 AI 宠物玩法,使用用 AI 生成超萌动物(水獭),做成宠物互动的场景式,这种视频播…

掌握Go语言的基本数据类型:编程世界的ABC

Go语言以其精炼的语法和卓越的性能,正日益成为现代软件开发的关键驱动力。在Go语言的众多特性中,基本数据类型的重要性不言而喻,它们是构建任何复杂应用程序的基石。正如掌握任何新语言需从其基础元素开始,Go语言的学习同样始于对…

【qt】windows下qt连接数据库

1.新建数据库 create database mydatabase;2.新建表结构,并插入数据 3.qt下连接数据库 1.连接数据库,需要加sql 2.添加QsqlDatabase头文件,使用提示句柄,头文件QMessageBox 3.连接数据库 4.界面设计 5.插入实现 注意这里如果…

盛水最多的容器问题

目录 一题目概述: 二思路理解: 三解答代码: 一题目概述: leetcode链接:. - 力扣(LeetCode) 二思路理解: 首先这道题虽然看起来简单,很难不让人想到直接暴力破解它&a…

【操作系统】12.文件系统是怎么管理的?

2.文件系统是怎么管理的? 2.1 文件系统的实现 2.1.1 文件系统层次结构 用户调用接口 文件目录系统 存取控制验证模块 逻辑文件系统与文件信息缓冲区 物理文件系统 辅助分配模块 设备管理程序模块 2.1.2 目录实现 线性列表 哈希表 2.1.3 文件实现 文件分配方式 连续…

Unity射击游戏开发教程:(32)控制子弹射速

显示发射速度从较慢到较快的变化 这是一篇关于如何实现射速以防止玩家乱按射击按钮的快速文章。 通常做法 说到要控制子弹的射速,可以通过以下几种方法: 通过设置子弹的发射间隔时间来控制射速。在每次子弹发射后,设置一个时间延迟,确保下一枚子弹不能立即发射。 利用计时…

Redis下载安装使用教程图文教程(超详细)

《网络安全自学教程》 Redis下载安装使用 1、下载安装2、基础配置2.1、远程连接2、保护模式3、默认密码4、后台运行4.1 验证服务是否启动4.2 停止服务 1、下载安装 1)到Redis中文网 https://www.redis.com.cn/download.html 下载Redis安装包 2)Redis的安…

市场动荡!宝马「赶超」特斯拉?华为打响高阶智驾「价格战」

新能源汽车市场正在进入动荡期。 本周,欧洲市场传来重磅消息,宝马在欧洲的电动汽车销量首次超过特斯拉。根据JATO Dynamics的数据,7月宝马在欧洲本土市场销售了14,869辆电动汽车,以308辆的微弱优势首次超过特斯拉。 数据显示&…

“蓝屏事件”阴魂不散,微软安全更新导致 Linux 系统无法启动即解决方案

最近,众多Linux用户报告称他们的设备在尝试启动时,收到了一条神秘的错误消息:“系统出了严重问题。”这起事件的罪魁祸首是微软在月度安全更新中发布的一个补丁,用于修复一个存在已久的GRUB漏洞。 这次更新却导致了Linux和Windows…

Adobe InDesign(ID)专业的排版设计软件win/mac软件下载安装和快捷键的使用

一、Adobe InDesign软件概述 1.1 软件简介 Adobe InDesign是Adobe Systems公司开发的一款面向印刷和数字媒体出版物的专业排版设计软件。它广泛应用于书籍、杂志、报纸、宣传册、海报、广告、电子书等多种出版物的设计与制作中。InDesign以其强大的排版功能、灵活的页面布局工…

7 款最佳软件,可恢复从回收站中删除的文件

您是否曾经从回收站中删除了文件并想要恢复它们?这似乎是不可能的。 但是,从技术上讲,永久删除的文件不会从您的硬盘驱动器存储中删除,而是更改为无法识别的数据并被您的新文件覆盖。 这使您能够从系统中恢复永久删除的文件。 …

Verdi中快速查找define宏定义及其active/inactive状态

可以通过如下方式查看: 打开verdi后 通过 菜单栏tools -> vc apps toolbox 在VC apps toolbox 中选择Utility-》macro utility : 用macro utility中的三个工具分别打印报告(各自针对不同的宏)。生成的log文件中会显示宏定义的…

VMware上Linux系统报错--mount: 在 /dev/sr0 上找不到媒体【解决办法】

问题背景 在VMware上使用CentOS7系统,想要在CentOS7系统上挂载一个镜像文件,在系统下使用mount命令挂载报错如下所示: 解决办法 用lsblk命令查看sr0设备是存在的 查看虚拟机侧问题,右键虚拟机,点击设置 镜像文件已选…

宝塔面板部署雷池社区版教程

宝塔面板部署雷池社区版教程 简单介绍一下宝塔面板,安全高效的服务器运维面板,使用宝塔面板的人非常多 在网站管理上,许多用户都是通过宝塔面板进行管理,宝塔面板的Nginx默认监听端口为80和443,这就导致共存部署时雷…

AIGC大模型应用教学平台

1、基本介绍 AIGC大模型应用技术实训平台是我公司针对高校开设的《AIGC大模型应用技术》课程配套的实训平台。该平台采用最新的AI推理加速芯片,提供30Tops算力和32GB存储,能够运行常见的大语言模型如ChatGLM2、Llama2、Baicuan、Mistral等,也…

vue2 part3

vscode安装vue3 snippets 天气案例 <!DOCTYPE html> <html><head><meta charset"UTF-8" /><title>姓名案例_插值语法实现</title><!-- 引入Vue --><script type"text/javascript" src"../js/vue.js&q…

【Linux】分析一段oom及oops报错日志

oom相关日志分析: Oom-killer错误是因系统内存分配不足&#xff0c;为保障系统正常运行会随机kill掉占用较多的内存进程。 该日志已经输出内存占满相关提示&#xff0c;内存上限为16G&#xff0c;当前已使用16G&#xff0c;内存限制导致分配失败次数为586755次。 OOPS相关日志…

24年最新最详细Python安装教程,附安装包!感觉收藏!

Python由荷兰国家数学与计算机科学研究中心的吉多范罗苏姆于1990年代初设计&#xff0c;作为一门叫做ABC语言的替代品。 Python提供了高效的高级数据结构&#xff0c;还能简单有效地面向对象编程。Python语法和动态类型&#xff0c;以及解释型语言的本质&#xff0c;使它成为多…

centos7解决病毒入侵 getty

首先使用top命令查看 找到文件地址 查看是否有自启动服务 关闭、停止、删除 tmp 病毒文件删除 清除标记 [roothost-192-168-0-66 bin]# chattr -ia /tmp/newsvc.sh [roothost-192-168-0-66 bin]# chattr -ia /tmp/redis2 [roothost-192-168-0-66 bin]# chattr -ia /tmp/svc* [r…

《黑神话:悟空》WeGame标准版和豪华版有啥区别?

黑神话悟空标准版和豪华版有什么区别&#xff1f;码笔记&#xff1a;标准版价格268元只是完整游戏&#xff0c;豪华版价格328在标准版的基础上另外提供兵器铜云棒、披挂百戏傩面、百戏护手、百戏衬钱衣、百戏吊腿、珍玩&#xff1a;风铎和游戏音乐精选集&#xff0c;详细如下&a…