Spring Boot解决循环依赖的过程详细记录

 更新时间:2024年09月12日 08:29:06   作者:从零开始的-CodeNinja之路  
这篇文章主要介绍了Spring Boot解决循环依赖的过程,Spring框架通过三级缓存机制解决循环依赖问题,分别为singletonObjects、earlySingletonObjects和singletonFactories,需要的朋友可以参考下

循环依赖

一、前言

环依赖是指两个或者多个bean互相依赖对方,从而形成一个闭环。例如:Bean A依赖于Bean B,而Bean B又依赖于Bean A。可能会导致Spring在尝试创建这些bean实例时出现问题,因为他们互相等待对方被创建,最终导致应用程序无法启动。

Spring是如何发现这种循环依赖的问题的呢?
通过依赖图来检测和发现循环依赖问题。如下步骤:

二、Bean的创建过程

Spring容器在启动时,会扫描配置文件(appliactionContext.xml)或者注解定义的bean,并且尝试创建这些bean的实例。创建bean实例的过程如下

  • 实例化:创建bean的实例。
  • 属性填充:为bean注入依赖其他的bean。
  • 初始化:执行自定义的初始化方法。
  • 依赖注入过程

在属性填充阶段,Spring会为每个bean注入他所依赖的bean。在这个过程中,Spring会跟踪哪些bean正在被创建,以便检测循环依赖。

三、 循环依赖检测机制

Spring通过一个名为“DefaultSingletonBeanRegistry”的类来跟踪单例bean的创建状态。该类维护了三个主要的缓存来管理bean的创建过程。

  • singletonObjects:一级缓存(存储完全初始化好的bean)
  • earlySingletonObjects:二级缓存(存储早期暴露的单例bean,用于解决循环依赖)
  • singletonFactories:三级缓存(存储用于创建bean实例的工厂)

具体的检测步骤

3.1 实例化阶段

当Spring开始实例化一个bean时,它会将这个bean标记为正在创建。这一步是通过将bean名称添加到一个“正在创建中的bean”集合(‘singletonCurrentlyInCreaontion’)中来实现的。

3.2 属性填充阶段

在属性填充阶段,Sping会为该bean注入其依赖的其他的bean。此时Spring会检查这个“其他的bean”是否已经在创建过程中。

如果依赖的bean已经在创建中,Spring会检测到循环依赖,并根据不同的注入方式采取不同的处理方式。
如果是构造函数注入,Spring会抛出‘BeanCurrentlyInCreationException’,因为无法解决构造函数中注入的循环依赖。

如果是Setter注入,Spring会从‘earlySingletonObjects’或‘singletonFactories’中获取依赖的bean,提前暴露一个部分创建的bean引用来解决循环依赖。

举例:

如Bean A和Bean B循环依赖

@Component
public class A {
    @Autowired
    private B b;

    public A() {
        System.out.println("A is created");
    }
}

@Component
public class B {
    @Autowired
    private A a;

    public B() {
        System.out.println("B is created");
    }
}

3.3 依赖注入

Spring的依赖注入过程:

  • 实例化Bean A

将Bean A 标记为正在创建,并添加到‘singletonCurrentlyInCreation’集合中。
实例化Bean A,并将其放入到三级缓存‘singletonFactories’中。(三级缓存存放的是创建实例化的bean工厂)

  • 填充Bean A的属性

此时发现Bean A依赖Bean B ,于是开始创建Bean B。
将Bean B标记为正在创建,并放入到‘singletonCurrentlyInCreation’集合中。
实例化Bean B,并将其放入到三级缓存‘singletonFactories’中。

  • 填充Bean B的属性

发现Bean B依赖于Bean A 。此时,检查到Bean A 已经在创建过程中,因此发现了循环依赖。
由于是Setter方法注入,Spring会从三级缓存‘singletonFactories’中获取一个部分创建的Bean A实例,提前暴露出来,放入二级缓存‘earlySingletonObjects’中。
使用这个部分创建出来的Bean A实例 来填充Bean B的属性。

  • 完成Bean B的创建

Bean B中的所有属性填充完毕后,Spring将Bean B的实例从三级缓存‘singletonFactories’中移到一级缓存‘singletonObjects’完全初始化好的bean中。

  • 回到Bean A的创建

继续为Bean A填充属性,此时可以从一级缓存‘singletonObejcts’中获取到完整的Bean B实例。
完成Bean A的创建,并将其从三级缓存‘singletonFactories’中移到一级缓存‘singletonObjects’中。

总结

到此这篇关于Spring Boot解决循环依赖的文章就介绍到这了,更多相关Spring Boot循环依赖解决内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • SpringBoot+Mybatis plus实现多数据源整合的实践

    SpringBoot+Mybatis plus实现多数据源整合的实践

    本文主要介绍了SpringBoot+Mybatis plus实现多数据源整合的实践,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-10-10
  • Java中ArrayList类的源码解析

    Java中ArrayList类的源码解析

    本文主要介绍了Java中ArrayList类的源码解析,具有很好的参考价值。下面跟着小编一起来看下吧
    2017-03-03
  • 关于Java中如何实现文件的读写操作

    关于Java中如何实现文件的读写操作

    在Java中,可以使用File和FileInputStream、FileOutputStream、BufferedReader、PrintWriter等类来进行文件读写操作,需要的朋友可以参考下
    2023-05-05
  • java警告:源发行版17 需要目标发行版17问题及解决

    java警告:源发行版17 需要目标发行版17问题及解决

    文章介绍了如何解决项目JDK版本不一致的问题,包括修改Project Structure、Modules、Dependencies和Settings中的JDK版本,以及在pom.xml中指定JDK源版本
    2024-11-11
  • 使用SpringMVC的@Validated注解验证的实现

    使用SpringMVC的@Validated注解验证的实现

    这篇文章主要介绍了使用SpringMVC的@Validated注解验证的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-08-08
  • Spring Boot 整合 JWT的方法

    Spring Boot 整合 JWT的方法

    这篇文章主要介绍了Spring Boot 整合 JWT的方法,文中实例代码非常详细,帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-07-07
  • JAVA Frame 窗体背景图片,首位相接滚动代码实例

    JAVA Frame 窗体背景图片,首位相接滚动代码实例

    这篇文章主要介绍了JAVA Frame 窗体背景图片,首位相接滚动代码示例,需要的朋友可以参考下复制代码
    2017-04-04
  • spring boot整合CAS配置详解

    spring boot整合CAS配置详解

    这篇文章主要介绍了spring boot整合CAS配置详解,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-05-05
  • java版微信公众平台消息接口应用示例

    java版微信公众平台消息接口应用示例

    这篇文章主要介绍了java版微信公众平台消息接口应用,结合实例形式对比分析了PHP与java应用微信公众平台接口的相关调用与操作技巧,需要的朋友可以参考下
    2017-07-07
  • maven中自定义MavenArchetype的实现

    maven中自定义MavenArchetype的实现

    Maven自身提供了许多Archetype来方便用户创建Project,为了避免在创建project时重复的拷贝和修改,我们通过自定义Archetype来规范显得还蛮有必要,下面就来介绍一下,感兴趣的可以了解一下
    2025-01-01

最新评论