Spring循环依赖详解

一、什么是循环依赖

在Spring框架中,循环依赖是指两个或更多的Bean相互间直接或间接地依赖对方的情况。这种依赖关系可能导致Bean的实例化过程中出现问题,但Spring通过其内部的处理机制能够解决某些类型的循环依赖。

简单的说就是A依赖B,B依赖C,C依赖A这样就构成了循环依赖。

二、循环依赖的3种情况

1. 构造器循环依赖

当Bean通过构造器参数注入其他Bean,并且存在循环依赖时,Spring无法解决这种依赖。这是因为Bean的实例化需要依赖于尚未创建完成的其他Bean,从而导致死锁。

复制代码
public class StudentA {

    private final StudentB studentB;

    public StudentA(StudentB studentB) {
        this.studentB = studentB;
    }
}


public class StudentB {

    private final StudentA studentA ;

    public StudentB(StudentA studentA) {
        this.studentA = studentA;
    }
}
2. 原型(prototype)模式下的setter注入循环依赖

循环依赖双方 scope 都是 prototype 的话,也会循环依赖失败,代码如下:

复制代码
@Service
@Scope("prototype")
public class AService {
    @Autowired
    BService bService;
}

@Service
@Scope("prototype")
public class BService {
    @Autowired
    AService aService;
}

两者的区别是:有状态的bean都使用Prototype作用域,无状态的一般都使用singleton单例作用域。

scope 为 prototype 意思就是每次请求都会创建一个新的实例对象,不用缓存里的。那么 AService 需要 BService,所以就去现场创建 BService,结果 BService 又需要 AService,继续现场创建,AService 又需要 BService...,所以最终就陷入到死循环了。

报错信息:

3. 单例(singleton)模式下的setter注入循环依赖

以上两种的循环依赖Spring并没有解决,需要开发者进行相应的设计调整或使用其他的依赖注入策略。但是单例模式下的setter注入循环依赖,Spring能够解决。

Spring的Bean实例化过程中会使用一个三级缓存机制:

  • 一级缓存(singletonObjects):存储已经完全初始化的单例Bean。
  • 二级缓存(earlySingletonObjects):存储尚未完全初始化的Bean,即已经实例化但尚未完成属性填充的Bean。
  • 三级缓存(singletonFactories):存储Bean的工厂对象,该对象可以创建Bean的实例并将其放置在二级缓存中。

当Spring检测到循环依赖时,它不会等待所有依赖都初始化完成,而是将Bean的工厂对象提前暴露给其他正在创建的Bean。这样,即使Bean尚未完成初始化,其他Bean也能引用它,从而打破了死锁状态。

懒加载解决循环依赖:

在某些情况下,可以使用****@Lazy****注解来延迟Bean的初始化,从而避免循环依赖

public class BeanA {
@Autowired
@Lazy
private BeanB beanB;
}

public class BeanB {
@Autowired
private BeanA beanA;
}

三、三级缓存

Spring能够轻松的解决属性的循环依赖正式用到了三级缓存,在AbstractBeanFactory中有详细的注释。

  • 一级缓存:singletonObjects,存放完全实例化属性赋值完成的Bean,直接可以使用。
  • 二级缓存:earlySingletonObjects,存放早期Bean的引用,尚未属性装配的Bean
  • 三级缓存:singletonFactories,三级缓存,存放实例化完成的Bean工厂。
1.getSingleton(beanName):源码如下:

从源码可以得知,doGetBean最初是查询缓存,一二三级缓存全部查询,如果三级缓存存在则将Bean早期引用存放在二级缓存中并移除三级缓存。(升级为二级缓存)

2. addSingletonFactory:源码如下

从源码得知,Bean在实例化完成之后会直接将未装配的Bean工厂存放在三级缓存中,并且移除二级缓存

3. addSingleton:源码如下:

Bean添加到一级缓存,移除二三级缓存。

相关推荐
2601_961194024 小时前
2026初级会计实务公式总结大全|计算题公式手册PDF
java·spring·eclipse·pdf·tomcat·hibernate
phltxy7 小时前
Spring AI 从提示词到多模态
java·人工智能·spring
LIUAWEIO9 小时前
vue里面下载配置使用zepto vue中怎样使用zepto
javascript·vue.js·es6·zepto
lantian9 小时前
TypeScript 三斜线指令完全指南:从入门到理解为什么不再需要它
前端·javascript·vue.js
用户938515635079 小时前
从"用栈实现队列"说起:深入理解 JavaScript 原型式面向对象
javascript
ZengLiangYi9 小时前
AI 编程工具的数据格式为什么不能统一
javascript·后端·架构
陈_杨9 小时前
鸿蒙APP开发-带你走进旧物集的时间线与收藏管理
前端·javascript
尼斯湖皮皮怪9 小时前
iceCoder双模详解
javascript
小雨下雨的雨9 小时前
月相分析工具鸿蒙PC Electron框架技术实现详解
前端·javascript·华为·electron
布依前端9 小时前
基于 Vue 3 的 Tiptap 富文本编辑器实践:tiptap-editor-vue3 项目介绍
前端·javascript·vue.js