Spring 循环依赖解析与解决方案

文章目录

    • 1. 什么是循环依赖?
      • 1.1 概念解析
      • 1.2 示例代码
    • 2. 循环依赖的类型
      • 2.1 构造器循环依赖(不可解决 ❌)
      • 2.2 Setter 方式或 `@Autowired` 方式的循环依赖(可解决 ✅)
    • 3. 解决循环依赖的方式
      • 3.1 方式一:使用 `@Lazy`(延迟加载) 🚀
      • 3.2 方式二:修改依赖关系 🛠️
      • 3.3 方式三:使用 `@PostConstruct` 进行初始化 🎯
      • 3.4 方式四:使用 `ApplicationContextAware` 直接获取 Bean ⚙️
    • 4. 结论与最佳实践 🏆

1. 什么是循环依赖?

1.1 概念解析

在 Spring 容器中,循环依赖(Circular Dependency) 是指:

  • Bean A 依赖 Bean B
  • Bean B 依赖 Bean A
  • 这种情况下,Spring 在创建 Bean 时会出现互相等待,可能导致异常。

1.2 示例代码

@Component
public class A {private final B b;@Autowiredpublic A(B b) {this.b = b;}
}@Component
public class B {private final A a;@Autowiredpublic B(A a) {this.a = a;}
}

💥 错误信息:

Caused by: org.springframework.beans.factory.BeanCurrentlyInCreationException:
Error creating bean with name 'A': Requested bean is currently in creation:
Is there an unresolvable circular reference?

2. 循环依赖的类型

2.1 构造器循环依赖(不可解决 ❌)

  • 通过构造方法互相依赖,Spring 无法提前暴露 Bean 实例,因此会直接报错。
  • 示例:(上面代码就属于此类)

2.2 Setter 方式或 @Autowired 方式的循环依赖(可解决 ✅)

  • Spring 默认支持这种循环依赖(单例模式下)。
  • 示例:
@Component
public class A {private B b;@Autowiredpublic void setB(B b) {this.b = b;}
}@Component
public class B {private A a;@Autowiredpublic void setA(A a) {this.a = a;}
}

📌 Spring 解决方式:

  • 一级缓存(singletonObjects): 存放完全实例化的 Bean。
  • 二级缓存(earlySingletonObjects): 存放提前暴露的半成品 Bean。
  • 三级缓存(singletonFactories): 存放创建 Bean 的工厂,以支持代理(如 AOP 代理)。

3. 解决循环依赖的方式

3.1 方式一:使用 @Lazy(延迟加载) 🚀

  • 只在需要时才创建 Bean,避免初始化时发生循环依赖。
@Component
public class A {private final B b;@Autowiredpublic A(@Lazy B b) {this.b = b;}
}

适用于构造器注入,但不能解决所有情况。

3.2 方式二:修改依赖关系 🛠️

  • 重新设计 Bean 关系,避免直接互相依赖。
  • 可以通过 引入中间层(如事件监听模式、消息队列等) 解决。

3.3 方式三:使用 @PostConstruct 进行初始化 🎯

  • 让 Spring 先创建 Bean,然后在初始化时再注入依赖。
@Component
public class A {private B b;@Autowiredpublic A() {}@PostConstructpublic void init() {this.b = SpringContextUtil.getBean(B.class);}
}

避免构造方法循环依赖,适用于某些场景。

3.4 方式四:使用 ApplicationContextAware 直接获取 Bean ⚙️

@Component
public class A implements ApplicationContextAware {private B b;private ApplicationContext applicationContext;@Overridepublic void setApplicationContext(ApplicationContext applicationContext) throws BeansException {this.applicationContext = applicationContext;}@PostConstructpublic void init() {this.b = applicationContext.getBean(B.class);}
}

适用于特定场景,但不推荐滥用 ApplicationContext,影响 Spring 设计模式。

4. 结论与最佳实践 🏆

方案适用情况备注
@Lazy适用于构造器循环依赖延迟加载可能会影响性能
修改依赖结构推荐最优方案,从根本上解决问题
@PostConstruct适用于部分场景但可能导致依赖获取不直观
ApplicationContext 获取 Bean可行但不推荐破坏 Spring 设计模式

🎯 最佳实践:

  1. 避免构造器循环依赖,优先使用 @Autowired 方式。
  2. 重构业务逻辑,减少不必要的循环依赖。
  3. 若无法避免,尝试 @Lazy@PostConstruct 方案。
  4. **尽量不使用 ApplicationContextAware,防止耦合过高。 **

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

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

相关文章

Cesium@1.126.0,创建3D瓦片,修改样式

第一步:添加3D建筑 Cesium.createOsmBuildingsAsync()这是一个异步方法,所以要写在一个异步函数里 创建一个函数 const create3DBuilding async (viewer) > {try {// 添加3D建筑const tileset await Cesium.createOsmBuildingsAsync();viewer.scen…

力扣-贪心-1005 k次取反后最大化的数组和

思路 找到绝对值最大的,然后如果是负数就变成正的,所有数遍历完之后,有两种情况,一种是k已经为0了,不需要再取反了,一种是所有数都为正数,k不为0,此时对绝对值最小的数操作即可 代…

vue2项目打包后js文件过大, 首次加载缓慢

vue2项目打包后js文件过大, 首次加载缓慢 安装插件 npm i compression-webpack-plugin6.1.1 -D配置vue.config.js const CompressionWebpackPlugin require(compression-webpack-plugin)module.exports {configureWebpack: {plugins:[new CompressionWebpackPlugin({filen…

高级SQL技术在Python项目中的应用:ORM与深度性能优化

引言 在现代Python项目开发中,数据库交互远不止是数据的简单存取,它已成为构建高性能、可维护应用的核心瓶颈和关键能力所在。 仅仅依赖基础SQL查询,虽然入门简单,却难以应对日益增长的应用挑战。这些挑战主要体现在以下几个方面: 性能瓶颈: 数据量剧增: 从百万到数十亿乃…

基于 C++ Qt 的 Fluent Design 组件库 QFluentWidgets

简介 QFluentWidgets 是一个基于 Qt 的 Fluent Designer 组件库,内置超过 150 个开箱即用的 Fluent Designer 组件,支持亮暗主题无缝切换和自定义主题色。 编译示例 以 Qt5 为例(Qt6 也支持),将 libQFluentWidgets.d…

抖音视频如何下载保存去水印

随着短视频平台的兴起,抖音作为国内最受欢迎的短视频平台之一,吸引了大量用户上传和观看各种创意视频。许多用户在浏览抖音视频时,往往会想要保存一些有趣或精彩的视频片段,但抖音视频通常会有水印,影响观看体验。为了…

React 源码揭秘 | 更新队列

前面几篇遇到updateQueue的时候,我们把它先简单的当成了一个队列处理,这篇我们来详细讨论一下这个更新队列。 有关updateQueue中的部分,可以见源码 UpdateQueue实现 Update对象 我们先来看一下UpdateQueue中的内容,Update对象&…

[SQL] 事务的四大特性(ACID)

🎄事务的四大特性 以下就是事务的四大特性,简称ACID。 原子性📢事务时不可分割的最小操作单元,要么全部成功,要么全部失败。一致性📢事务完成后,必须使所有的数据都保持一致隔离性&#x1f4e2…

DeepSeek 提示词:基础结构

🧑 博主简介:CSDN博客专家,历代文学网(PC端可以访问:https://literature.sinhy.com/#/?__c1000,移动端可微信小程序搜索“历代文学”)总架构师,15年工作经验,精通Java编…

如何使用 Python 连接 MySQL 数据库?

在Python开发中,连接MySQL数据库是一个常见的需求。 我们可以使用多种库来实现这一功能,其中最常用的是mysql-connector-python和PyMySQL。 下面我将详细介绍如何使用这两个库来连接MySQL数据库,并提供一些实际开发中的建议和注意事项。 1…

Apache DolphinScheduler系列1-单节点部署及测试报告

文章目录 整体说明一、部署环境二、版本号三、部署方案四、部署步骤4.1、上传部署包4.2、创建外部数据库4.3、修改元数据库配置4.4、上传MySQLl驱动程序4.5、初始化外部数据库4.6、启停服务4.7、访问页面五、常见问题及解决方式5.1、时间不一致5.2、异常终止5.3、大量日志5.4、…

LLM之论文阅读——Context Size对RAG的影响

前言 RAG 系统已经在多个行业中得到广泛应用,尤其是在企业内部文档查询等场景中。尽管 RAG 系统的应用日益广泛,关于其最佳配置的研究却相对缺乏,特别是在上下文大小、基础 LLM 选择以及检索方法等方面。 论文原文: On the Influence of Co…

人工智能(AI):科技新纪元的领航者

摘要 人工智能(AI)作为当今科技领域最具变革性的力量之一,正以惊人的速度重塑着我们的世界。本文旨在全面且专业地介绍人工智能,涵盖其定义、发展历程、关键技术、应用领域、面临的挑战以及未来展望等方面,以期为读者…

如何防止 Docker 注入了恶意脚本

根据您的描述,攻击者通过 CentOS 7 系统中的 Docker 注入了恶意脚本,导致自动启动名为 “masscan” 和 “x86botnigletjsw” 的进程。这些进程可能用于网络扫描或其他恶意活动。为了解决这一问题,建议您采取以下步骤: 1. 停止并删…

LLaMA-Factory|微调大语言模型初探索(4),64G显存微调13b模型

上篇文章记录了使用lora微调deepseek-7b,微调成功,但是微调llama3-8b显存爆炸,这次尝试使用qlora微调HQQ方式量化,微调更大参数体量的大语言模型,记录下来微调过程,仅供参考。 对过程不感兴趣的兄弟们可以直…

详解 Spring 配置数据源的两种方式

在 Spring 框架中配置数据源(DataSource)主要有两种方式: 通过 Setter 注入配置数据源通过 jdbc.properties 配置文件方式 本博文将使用 Druid 作为数据源,其在 Spring 项目中常见且高效。 Druid 被广泛认为是性能最佳的连接池…

项目进度管理工具:甘特图与关键路径法(2025实战指南)

在全球数字化转型加速的背景下,项目延期率高达42%的现状倒逼管理者掌握科学的进度管理工具。本文结合2025年最新实践,深度解析甘特图与关键路径法的原理及应用,助你构建精准可控的项目进度管理体系。 一、双剑合璧:工具组合的价值…

RAGS评测后的数据 如何利用influxdb和grafan 进行数据汇总查看

RAGS(通常指相关性、准确性、语法、流畅性)评测后的数据能借助 InfluxDB 存储,再利用 Grafana 进行可视化展示,实现从四个维度查看数据,并详细呈现每个问题对应的这四个指标情况。以下是详细步骤: 1. 环境准备 InfluxDB 安装与配置 依据自身操作系统,从 InfluxDB 官网下…

详解Redis如何持久化

引言 本文介绍了 Redis 的两种持久化方式:RDB 和 AOF。RDB 按时间间隔快照存储,AOF 记录写操作。阐述了它们的配置、工作原理、恢复数据的方法、性能与实践建议,如降低 fork 频率、控制内存等,还提到二者可配合使用,最…

HarmonyOS Design 介绍

HarmonyOS Design 介绍 文章目录 HarmonyOS Design 介绍一、HarmonyOS Design 是什么?1. 设计系统(Design System)2. UI 框架的支持3. 设计工具和资源4. 开发指南5. 与其他设计系统的对比总结 二、HarmonyOS Design 特点 | 应用场景1. Harmon…