当前位置: 首页 > news >正文

Spring是如何解决bean循环依赖的问题的

在Spring框架中,循环依赖是指两个或多个Bean之间相互依赖,形成了一个闭环的依赖关系。当存在循环依赖时,Bean的创建过程会陷入死循环,导致应用程序无法启动或出现异常。

说到循环依赖,首先我先说说bean的三级缓存

在Spring框架中,Bean的三级缓存是指在Bean的创建过程中,Spring容器为了提高性能而采用的一种缓存机制。这个缓存机制包括三个级别的缓存,分别是singletonObjects、earlySingletonObjects和singletonFactories。

  1. singletonObjects:这是最终的单例对象缓存。当一个Bean的创建过程完成后,它会被放入singletonObjects缓存中。以后如果再次请求获取该Bean,Spring将直接从singletonObjects缓存中返回已经创建好的实例。

  2. earlySingletonObjects:这是早期的单例对象缓存。在Bean的创建过程中,当依赖关系解析完成但还未完成初始化时,Bean会被放入earlySingletonObjects缓存中。这样,当其他Bean需要引用该Bean时,可以提前获取到它的引用,避免循环依赖的问题。

  3. singletonFactories:这是用于存储Bean工厂的缓存。当Bean正在创建过程中,它的工厂方法会被放入singletonFactories缓存中。这样,在工厂方法被调用时,可以直接从缓存中获取工厂实例,而不需要重新创建。

通过使用三级缓存,Spring能够有效地管理Bean的创建过程和依赖关系,提高应用程序的性能和效率。这种缓存机制在单例作用域的Bean中特别有用,因为单例Bean在整个应用程序中只会被创建一次,而且可能存在循环依赖的情况。

需要注意的是,三级缓存的使用是为了提高性能和解决循环依赖问题,而不是为了缓存大量的Bean实例。因此,对于非单例作用域的Bean,Spring不会使用三级缓存,而是每次请求都会创建一个新的实例。

看完了三级缓存,大概都知道循环依赖怎么解决的,那我们来具体看一下策略吧

为了解决循环依赖问题,Spring使用了三级缓存和"提前暴露对象"的策略。下面是Spring解决循环依赖问题的大致过程:

  1. 创建Bean对象:当Spring容器需要创建一个Bean时,会先检查singletonObjects缓存中是否已经存在该Bean的实例。如果存在,则直接返回实例;如果不存在,则继续下一步。

  2. 提前暴露对象:在Bean的创建过程中,当依赖关系解析完成但还未完成初始化时,Spring会将当前Bean实例提前暴露给其他正在创建的Bean。这样,其他Bean可以提前获取到该Bean的引用,避免循环依赖的问题。

  3. 三级缓存处理:当发现循环依赖时,Spring会使用三级缓存来处理。首先,将正在创建的Bean工厂方法放入singletonFactories缓存中。然后,继续创建Bean的依赖关系,并将已经解析完成的依赖放入earlySingletonObjects缓存中。最后,完成Bean的创建并放入singletonObjects缓存中。

  4. 依赖注入:当所有Bean的创建过程完成后,Spring会执行依赖注入操作,将Bean的依赖关系注入到对应的属性中。

通过以上步骤,Spring能够解决循环依赖问题。它通过提前暴露对象和三级缓存的机制,在Bean的创建过程中合理处理依赖关系,确保循环依赖时能够正确地获取到已经创建好的实例或工厂方法,从而避免死循环的情况发生。

需要注意的是,Spring对于构造函数注入的循环依赖无法解决,因为在构造函数阶段无法提前暴露对象。对于这种情况,可以考虑使用Setter方法注入或通过引入一个中间层来解决循环依赖。

http://www.lryc.cn/news/212622.html

相关文章:

  • [移动通讯]【Carrier Aggregation-9】【 Radio Resource Control (RRC) Aspects】
  • 故障预测与健康管理(PHM)的由来以及当前面临的挑战
  • 【ChatGPT瀑布到水母】AI 在驱动软件研发的革新与实践
  • 【Django】项目模型
  • 字符集详解
  • Vert.x学习笔记-什么是Vert.x
  • AcWing 第127场周赛 构造矩阵
  • Seata入门系列【15】@GlobalLock注解使用场景及源码分析
  • Dubbo 路由及负载均衡性能优化
  • Python数据可视化入门指南
  • 我的ChatGPT的几个使用场景
  • 3 — NLP 中的标记化:分解文本数据的艺术
  • C++-类与对象(上)
  • 多进程间通信学习之无名管道
  • flink常用的几种调优手段的优缺点
  • 如何选择安全又可靠的文件数据同步软件?
  • 使用反射调用类的私有内部类的私有方法
  • 记一次 AWD 比赛中曲折的 Linux 提权
  • [SpringCloud] Feign 与 Gateway 简介
  • [Unity] 个人编码规范与命名准则参考
  • 堆栈与队列算法-以链表来实现队列
  • 快速入门:使用 Spring Boot 构建 Web 应用程序
  • 01.CentOS7静默安装oracle11g
  • SASE安全访问服务边缘
  • k8s集群升级
  • redis原理 主从同步和哨兵集群
  • 四季古诗赏析
  • 【网络协议】聊聊套接字socket
  • GEO生信数据挖掘(十一)STRING数据库PPI蛋白互作网络 Cytoscape个性化绘图【SCI 指日可待】
  • api接口更新钉钉文档