Spring三级缓存详解之循环依赖解决_Java理解Spring框架的底层机制

Spring通过三级缓存机制解决单例Bean的循环依赖问题,其中一级缓存(singletonObjects)存放完全初始化的Bean实例,二级缓存(earlySingletonObjects)存放早期引用的Bean实例,三级缓存(singletonFactories)存放用于生成早期引用的ObjectFactory;当Bean A创建过程中依赖Bean B,而Bean B又依赖Bean A时,Spring会将A的ObjectFactory放入三级缓存,B在需要A时通过该工厂获取A的早期引用(可能是代理对象),并将其放入二级缓存供B使用,待B初始化完成后,A继续初始化并最终将完整实例放入一级缓存,从而打破循环依赖;之所以需要三级缓存而非二级,是因为在涉及AOP代理时,若直接将原始对象放入二级缓存会导致B持有的是未代理的实例,而三级缓存的ObjectFactory能按需生成代理后的早期引用,确保AOP功能正确执行;Spring通过singletonsCurrentlyInCreation标识当前正在创建的Bean,当发现某Bean正在创建中又被请求时,即触发从三级缓存获取ObjectFactory并生成早期引用的机制,从而精准判断并解决循环依赖。

spring三级缓存详解之循环依赖解决_java理解spring框架的底层机制

Spring框架通过其精妙的三级缓存机制,巧妙且高效地解决了单例Bean之间的循环依赖问题。它并非简单地依靠一次性创建,而是在Bean的生命周期不同阶段,通过暴露不同状态的Bean实例,最终确保所有相互依赖的Bean都能被正确初始化并注入,避免了传统方式下因依赖无法满足而导致的创建死锁。这种设计是Spring底层机制中一个非常值得深入理解的亮点。

解决方案

Spring解决循环依赖的核心在于其三级缓存的设计与协同工作。这个过程可以概括为:当一个Bean A开始创建,并被实例化后(但尚未完成属性填充和初始化方法调用),它会先将自己包装成一个

ObjectFactory

放入第三级缓存。如果此时Bean A在后续的属性填充过程中需要依赖Bean B,而Bean B又反过来依赖Bean A,那么当Spring尝试创建Bean B并发现它需要Bean A时,它会首先从缓存中查找Bean A。由于Bean A的

ObjectFactory

已存在于第三级缓存,Spring就能通过这个工厂获取到Bean A的“早期引用”(一个尚未完全初始化的实例)。这个早期引用随后会被放入第二级缓存,供Bean B使用。这样,Bean B就能顺利完成初始化。当Bean B初始化完成后,Bean A可以继续它的初始化过程,填充属性(包括已经完整初始化好的Bean B),并最终完成自身初始化。一旦Bean A完全初始化,它的完整实例会替换掉第二级缓存中的早期引用,并最终被放入第一级缓存。

具体来说,这三级缓存分别是:

一级缓存(

singletonObjects

): 存放已经完全初始化并可供使用的单例Bean。这是最“成熟”的Bean。二级缓存(

earlySingletonObjects

): 存放早期暴露的Bean实例。这些Bean可能已经实例化但尚未完成属性填充或初始化方法调用,也可能已经经过了AOP代理。三级缓存(

singletonFactories

): 存放的是

ObjectFactory

,一个能够生产早期Bean实例的工厂。这个工厂在需要时才会被调用,用于获取Bean的早期引用,特别是在处理AOP代理时显得尤为重要。

Spring三级缓存具体指哪些,它们各自承担了什么职责?

在Spring的单例Bean生命周期管理中,这三层缓存各司其职,共同构筑了一个健壮的循环依赖解决方案。

立即学习“Java免费学习笔记(深入)”;

首先是

singletonObjects

,这是最核心的一级缓存,它直接映射了Bean名称到其完全初始化、可供使用的单例实例。可以把它想象成一个“成品仓库”,所有经过完整生命周期(实例化、属性填充、初始化方法执行)的Bean最终都会安家于此。当其他Bean需要依赖时,Spring会优先从这里查找。

接着是

earlySingletonObjects

,这是二级缓存,它扮演了一个“半成品仓库”的角色。当一个Bean被实例化后,但其属性尚未完全填充,或者初始化方法尚未执行时,它的早期引用(可能是原始对象,也可能是经过AOP处理的代理对象)就会被放置在这里。这个缓存的存在,是为了在循环依赖发生时,能够提供一个“临时”的、可用的Bean引用,让依赖它的Bean能够继续初始化。一旦这个Bean完成了所有初始化步骤,它就会从二级缓存中移除,并晋升到一级缓存。

最后,也是最关键的,是

singletonFactories

,这是三级缓存,它存储的不是Bean实例本身,而是一个

ObjectFactory

(对象工厂)。这个工厂的职责是“按需生产”Bean的早期引用。为什么需要一个工厂而不是直接存放早期实例呢?这主要是为了解决AOP代理的复杂性。如果一个Bean需要被AOP代理,那么它的早期引用应该是代理对象,而不是原始对象。这个

ObjectFactory

就封装了生成代理对象的逻辑。只有当其他Bean真正需要这个循环依赖的Bean时,Spring才会调用这个

ObjectFactory

来获取早期引用。这样,就避免了不必要的AOP代理创建,也确保了获取到的早期引用是正确的(即可能是代理后的)。

这三者协同工作,构成了一个巧妙的“生产线”。当A需要B,B需要A时,A实例化后,其

ObjectFactory

先入三级缓存。B需要A时,通过三级缓存的

ObjectFactory

拿到A的早期引用,放入二级缓存供B使用。B初始化完毕,A继续初始化,最终A的完整实例入一级缓存。这种机制确保了在Bean还未完全“成熟”时,也能提供一个可用的引用,从而打破循环依赖的僵局。

为什么需要三级缓存,二级缓存不能解决循环依赖吗?

这是一个非常经典的问题,也是理解Spring循环依赖解决机制深度的关键点。答案是:如果仅仅是纯粹的循环依赖(即没有AOP代理),二级缓存确实可以解决问题。但一旦涉及到AOP(面向切面编程),二级缓存就显得力不从心了,这就是三级缓存存在的根本原因。

想象一下这个场景:Bean A依赖Bean B,Bean B依赖Bean A,并且Bean A还需要被AOP代理。

如果只有一级和二级缓存:

Spring开始创建Bean A,实例化Bean A的原始对象。将Bean A的原始对象(早期引用)放入二级缓存

earlySingletonObjects

。Bean A在填充属性时需要Bean B,Spring开始创建Bean B。Bean B在填充属性时需要Bean A,从二级缓存

earlySingletonObjects

中获取到Bean A的原始对象。Bean B完成初始化。Bean A继续初始化,此时Spring会执行AOP代理逻辑,生成Bean A的代理对象。问题来了:Bean B持有的是Bean A的原始对象引用,而不是AOP代理后的引用。这导致Bean B调用的A的方法不会经过AOP切面,这显然是不正确的行为。

而三级缓存的引入,正是为了解决这个“AOP代理时机”的问题。三级缓存

singletonFactories

中存储的是一个

ObjectFactory

,这个工厂能够按需生成Bean的早期引用。这个“按需”和“生成”是关键。

当Bean A实例化后,Spring会将其原始对象包装成一个

ObjectFactory

放入三级缓存。这个

ObjectFactory

里面封装了生成AOP代理的逻辑(如果需要代理的话)。

当Bean B需要Bean A时,它会从三级缓存中获取到这个

ObjectFactory

,然后调用

getObject()

方法。此时,

getObject()

方法会判断Bean A是否需要AOP代理。如果需要,它就会立即生成Bean A的代理对象,并将这个代理对象作为早期引用返回给Bean B。这个代理对象随后会被放入二级缓存,同时从三级缓存中移除。

这样一来,Bean B获取到的就是Bean A的“正确”引用——一个可能已经被AOP代理过的对象。当Bean A最终完成所有初始化步骤时,它的一级缓存中的最终实例(也可能是代理对象)会替换掉二级缓存中的早期引用。整个过程中,所有Bean都持有了彼此的正确引用,包括AOP代理后的实例。

所以,三级缓存的真正价值在于它提供了一个“延迟生成AOP代理”的机制,确保了在循环依赖场景下,注入的早期引用是经过AOP处理的(如果需要),从而保证了AOP功能的正确性。没有它,AOP和循环依赖的结合就会出现问题。

Spring是如何判断并触发三级缓存机制来解决循环依赖的?

Spring在Bean的创建过程中,通过一套严谨的内部状态管理和查找逻辑来判断并触发三级缓存机制,从而解决循环依赖。这个过程主要发生在

DefaultSingletonBeanRegistry

类中的

getSingleton()

方法内部,这是Spring获取单例Bean的核心入口。

当Spring尝试获取一个单例Bean时(例如,通过

getBean()

方法),它会按照以下步骤进行:

尝试从一级缓存(

singletonObjects

)获取: 这是最直接的查找,如果Bean已经完全初始化并存在于此,直接返回。检查Bean是否正在创建中: 如果一级缓存中没有,Spring会检查

singletonsCurrentlyInCreation

这个Set。这个Set记录了当前正在创建过程中的Bean名称。如果发现当前请求的Bean正在创建中,这就意味着可能存在循环依赖。允许早期引用(

allowEarlyReference

): 对于单例Bean,Spring默认是允许早期引用的。如果Bean正在创建中且允许早期引用,Spring会进一步尝试从二级或三级缓存获取。尝试从二级缓存(

earlySingletonObjects

)获取: 如果Bean正在创建中,并且之前已经有其他Bean请求过它的早期引用,那么这个早期引用可能已经被放入了二级缓存。如果能找到,直接返回这个早期引用。尝试从三级缓存(

singletonFactories

)获取: 如果二级缓存中也没有,但Bean确实正在创建中,Spring会检查三级缓存。如果三级缓存中存在这个Bean对应的

ObjectFactory

,Spring就会调用这个

ObjectFactory

getObject()

方法。

getObject()

方法会负责生成Bean的早期引用。这个早期引用可能是原始的Bean实例,也可能是在AOP增强逻辑作用下生成的代理实例。一旦通过

ObjectFactory

获取到早期引用,Spring会立即将这个早期引用放入二级缓存

earlySingletonObjects

,并同时从三级缓存

singletonFactories

中移除对应的

ObjectFactory

。这是为了避免重复生成早期引用,也确保了后续对该Bean早期引用的请求直接从二级缓存获取,效率更高。注入并完成初始化: 获取到早期引用后,依赖它的Bean就可以继续完成属性填充和初始化。当这个早期引用所对应的Bean最终完成所有初始化步骤后,它的完整实例会晋升到一级缓存,替换掉之前在二级缓存中的早期引用。

这个机制的关键在于

singletonsCurrentlyInCreation

这个Set,它就像一个“正在施工”的标志牌。当Spring发现一个Bean正在施工中又被请求时,它就知道可能遇到了循环依赖,于是会启动从二级和三级缓存获取早期引用的逻辑。这种设计确保了只有在真正需要时才触发三级缓存的

ObjectFactory

,既解决了循环依赖,又兼顾了性能和AOP的正确性。

以上就是Spring三级缓存详解之循环依赖解决_Java理解Spring框架的底层机制的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/106398.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
苹果基带坏了有必要修吗?不修会有什么影响?
上一篇 2025年11月22日 08:22:49
《怪物猎人:荒野》更新完毕!9星高难怪、随机护石上线
下一篇 2025年11月22日 08:24:51

相关推荐

  • 修复Django电商项目中AJAX过滤产品列表图片不显示问题

    在Django电商项目中,当使用AJAX动态加载过滤后的产品列表时,常遇到图片无法正常显示的问题。这通常是由于前端模板中图片加载方式(如data-setbg属性结合JavaScript库)与AJAX动态内容更新机制不兼容所致。解决方案是直接在AJAX返回的HTML中使用标准的标签来渲染图片,确保浏览…

    2026年5月10日
    000
  • 修复点击时按钮抖动:CSS垂直对齐实践

    本文探讨了在Web开发中,交互式按钮(如播放/暂停按钮)在点击时发生意外垂直位移的问题。通过分析CSS样式变化对元素布局的影响,我们发现这是由于按钮不同状态下的边框样式和内边距改变,以及默认的垂直对齐行为共同作用所致。核心解决方案是利用CSS的vertical-align属性,将其设置为middle…

    2026年5月10日
    100
  • 理解编程指令:当结果正确,但实现方式不符要求时

    本文探讨了在编程实践中,即使程序输出了正确的结果,但若其实现方式未能严格遵循既定指令,仍可能被视为“不正确”的问题。我们将通过具体示例,对比直接求和与累加求和两种实现策略,强调理解和遵守编程规范的重要性,以确保代码的健壮性、可维护性及符合项目要求。 在软件开发过程中,我们经常会遇到这样的情况:编写的…

    2026年5月10日
    000
  • 使用 Jupyter Notebook 进行探索性数据分析

    Jupyter Notebook通过单元格实现代码与Markdown结合,支持数据导入(pandas)、清洗(fillna)、探索(matplotlib/seaborn可视化)、统计分析(describe/corr)和特征工程,便于记录与分享分析过程。 Jupyter Notebook 是进行探索性…

    2026年5月10日
    000
  • 《魔兽世界》将于6月11日开启国服回归技术测试

    《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试

    《%ign%ignore_a_1%re_a_1%》官方宣布,将于6月11日开启国服回归技术测试,时间为7天,并称可以在6月内正式开服,玩家们可以访问官网下载战网客户端并预下载“巫妖王之怒”客户端,技术测试详情见下图。 WordAi WordAI是一个AI驱动的内容重写平台 53 查看详情 以上就是《…

    2026年5月10日 用户投稿
    200
  • 如何在HTML中插入表单元素_HTML表单控件与输入类型使用指南

    HTML表单通过标签构建,包含action和method属性定义数据提交目标与方式,常用input类型如text、password、email等适配不同输入需求,配合label、required、placeholder提升可用性,结合textarea、select、button等控件实现完整交互,是…

    2026年5月10日
    100
  • 前端缓存策略与JavaScript存储管理

    根据数据特性选择合适的存储方式并制定清晰的读写与清理逻辑,能显著提升前端性能;合理运用Cookie、localStorage、sessionStorage、IndexedDB及Cache API,结合缓存策略与定期清理机制,可在保证用户体验的同时避免安全与性能隐患。 前端缓存和JavaScript存…

    2026年5月10日
    200
  • HTML5网页如何实现手势操作 HTML5网页移动端交互的处理技巧

    首先利用原生touch事件实现滑动判断,再通过preventDefault解决滚动冲突,接着引入Hammer.js处理复杂手势,最后通过优化点击区域、避免事件冲突和增加视觉反馈提升体验。 在移动端浏览器中,HTML5网页可以通过触摸事件实现手势操作,提升用户体验。虽然原生JavaScript提供了基…

    2026年5月10日
    000
  • 如何插入查询结果数据_SQL插入Select查询结果方法

    如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法

    使用INSERT INTO…SELECT语句可高效插入数据,通过NOT EXISTS、LEFT JOIN、MERGE语句或唯一约束避免重复;表结构不一致时可通过别名、类型转换、默认值或计算字段处理;结合存储过程可提升可维护性,支持参数化与动态SQL。 将查询结果数据插入到另一个表中,可以…

    2026年5月10日 用户投稿
    000
  • Discord.py 交互按钮超时与持久化解决方案

    本教程旨在解决Discord.py中交互按钮在一段时间后出现“This Interaction Failed”错误的问题。我们将深入探讨视图(View)的超时机制,并提供通过正确设置timeout参数以及利用bot.add_view()方法实现按钮持久化的具体方案,确保您的机器人交互功能稳定可靠,即…

    2026年5月10日
    000
  • JavaScript 闭包:理解闭包原理与内存泄漏问题

    闭包是函数访问其外部作用域变量的能力,即使外部函数已执行完毕。如 inner 函数引用 outer 中的 count,形成闭包,使变量持久存在。闭包本身无害,但可能因延长变量生命周期导致内存泄漏,例如事件监听器引用大对象时。若未及时清理 DOM 事件或定时器,闭包会阻止垃圾回收,造成内存占用过高。解…

    2026年5月10日
    100
  • JavaScript 动态菜单点击高亮效果实现教程

    本教程详细介绍了如何使用 JavaScript 实现动态菜单的点击高亮功能。通过事件委托和状态管理,当用户点击菜单项时,被点击项会高亮显示(绿色),同时其他菜单项恢复默认样式(白色)。这种方法避免了不必要的DOM操作,提高了性能和代码可维护性,确保了无论点击方向如何,功能都能稳定运行。 动态菜单高亮…

    2026年5月10日
    200
  • JS如何实现迭代器?迭代器协议

    JavaScript中实现迭代器需遵循可迭代协议和迭代器协议,通过定义[Symbol.iterator]方法返回具备next()方法的迭代器对象,从而支持for…of和展开运算符;该机制统一了数据结构的遍历接口,实现惰性求值,适用于自定义对象、树、图及无限序列等复杂场景,提升代码通用性与…

    2026年5月10日
    100
  • JavaScript函数中插入加载动画(Spinner)的正确方法

    本文旨在解决在JavaScript函数中插入加载动画(Spinner)时遇到的异步问题。通过引入async/await和Promise.all,确保在数据处理完成前后正确显示和隐藏加载动画,提升用户体验。我们将提供两种实现方案,并详细解释其原理和优势。 在Web开发中,当执行耗时操作时,显示加载动画…

    2026年5月10日
    100
  • 三星不再独享,消息称搭载骁龙 8 Gen 3 领先版处理器新机即将发布

    三星不再独享,消息称搭载骁龙 8 Gen 3 领先版处理器新机即将发布三星不再独享,消息称搭载骁龙 8 Gen 3 领先版处理器新机即将发布三星不再独享,消息称搭载骁龙 8 Gen 3 领先版处理器新机即将发布三星不再独享,消息称搭载骁龙 8 Gen 3 领先版处理器新机即将发布

    6 月 15 日消息,据博主@肥威 今日爆料,搭载骁龙 8 Gen 3 领先版%ign%ignore_a_1%re_a_1%的新机即将发布,把之前的 for Galaxy 改成“for Everybody”。 Pic Copilot AI时代的顶级电商设计师,轻松打造爆款产品图片 158 查看详情 …

    2026年5月10日 用户投稿
    100
  • 动态更新圆形进度条:JavaScript成绩计算器集成指南

    本文档旨在指导开发者如何将JavaScript成绩计算系统与动态圆形进度条集成,实现可视化展示平均成绩。我们将详细讲解如何修改现有的JavaScript代码,使其在计算出平均分后,能够动态更新圆形进度条的进度,从而提供更直观的用户体验。本文档包含详细的代码示例和注意事项,帮助开发者轻松实现这一功能。…

    2026年5月10日
    000
  • Golang使用Protobuf定义接口与消息格式

    Protobuf通过字段编号实现兼容性,新增字段可忽略、删除字段可保留编号,确保新旧版本互操作,支持服务独立演进。 在Golang项目中,利用Protobuf定义接口和消息格式,本质上是为服务间通信构建了一套高效、类型安全且跨语言的契约。它让数据结构清晰可见,RPC调用标准化,极大地简化了分布式系统…

    2026年5月10日
    000
  • JavaScript计算器开发:解决数值显示与初始化问题

    本教程深入探讨了使用JavaScript构建计算器时常见的数值显示异常问题,特别是由于类属性未初始化导致的`Cannot read properties of undefined`错误。我们将详细分析问题根源,并通过在构造函数中调用初始化方法来解决该问题,同时优化显示逻辑,确保计算器功能稳定且界面显…

    2026年5月10日
    000
  • 高通预热 2023 骁龙峰会:以AI为主题,10 月 25-26 日举行

    高通预热 2023 骁龙峰会:以AI为主题,10 月 25-26 日举行高通预热 2023 骁龙峰会:以AI为主题,10 月 25-26 日举行高通预热 2023 骁龙峰会:以AI为主题,10 月 25-26 日举行高通预热 2023 骁龙峰会:以AI为主题,10 月 25-26 日举行

    【环球网科技综合报道】10月17日消息,高通今日对 2023 骁龙峰会进行了预热,本次大会将以 %ign%ignore_a_1%re_a_1% 为主题,届时骁龙 8 gen 3 处理器也很大可能在本届峰会亮相。 在临近活动召开之日,相关业内人士也透露了高通骁龙8Gen3跑分及规格。据悉,高通骁龙8 …

    2026年5月10日 用户投稿
    000
  • 使用 Ajax 和 FormData 实现文件上传及文本数据提交的完整教程

    本文旨在解决在使用 Ajax 和 FormData 进行文件上传时,遇到的 $_POST 和 $_FILES 为空的问题。通过详细的代码示例和解释,我们将展示如何正确地构建 FormData 对象,并通过 Ajax 将文件和文本数据发送到服务器端,同时避免常见的错误配置,确保数据能够成功地被 PHP…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信