【Spring Boot】Spring Boot解决循环依赖
创始人
2024-11-14 10:06:32
0

目录

  • 循环依赖
    • 一、前言
    • 二、Bean的创建过程
    • 三、 循环依赖检测机制
      • 3.1 实例化阶段
      • 3.2 属性填充阶段
      • 3.3 依赖注入

循环依赖

一、前言

环依赖是指两个或者多个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的依赖注入过程:

  1. 实例化Bean A

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

  1. 填充Bean A的属性

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

  1. 填充Bean B的属性

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

  1. 完成Bean B的创建

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

  1. 回到Bean A的创建

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

相关内容

热门资讯

透视方针!德州局脚本,玉海楼茶... 透视方针!德州局脚本,玉海楼茶苑脚本(辅助)真是是有攻略(哔哩哔哩)1、操作简单,无需玉海楼茶苑脚本...
针对!顺欣茶楼辅助器,xpok... 针对!顺欣茶楼辅助器,xpoker辅助神器(透视)开挂辅助方法(哔哩哔哩)1、游戏颠覆性的策略玩法,...
五分钟绝活儿!wepoker透... 五分钟绝活儿!wepoker透视脚本下载,新九游辅助软件(辅助)一直是真的app(哔哩哔哩)wepo...
透视绝活儿!aapoker怎么... 透视绝活儿!aapoker怎么控制牌,AApoker辅助确实有脚本,第5分钟教程(今日头条)亲,关键...
透视指南书!约局吧如何查看是否... 透视指南书!约局吧如何查看是否有挂,闲娱江西脚本(辅助)一直存在有技巧(哔哩哔哩)1、超多福利:超高...
近期!填大坑辅助工具排行,拱趴... 近期!填大坑辅助工具排行,拱趴大菠萝辅助神器(透视)开挂辅助攻略(哔哩哔哩)1、该软件可以轻松地帮助...
一分钟举措!大菠萝挂机辅助,小... 一分钟举措!大菠萝挂机辅助,小程序财神十三张祈福(辅助)一贯真的有辅助器(哔哩哔哩)1、下载好小程序...
透视经验!wejoker开挂,... 透视经验!wejoker开挂,WePoKer正规确实真的有安装,4分钟教程(真是有挂)1、任何wej...
据权威媒体报道!新西游辅助器,... 据权威媒体报道!新西游辅助器,we poker免费辅助器(透视)开挂辅助脚本(哔哩哔哩)1、许多玩家...
九分钟秘籍!人海大厅挂件怎么买... 九分钟秘籍!人海大厅挂件怎么买,财神13章作必弊码大全(辅助)总是有挂插件(哔哩哔哩)1、每一步都需...