Spring循环依赖解决方法及原理

ops/2024/9/22 13:46:21/

在Spring框架中,循环依赖(circular dependency)是指两个或多个Bean相互依赖,形成一个环。Spring采用了多种方法来解决循环依赖问题,主要方法包括三级缓存和提前暴露Bean引用。以下是Spring解决循环依赖的原理和方法:

三级缓存

Spring使用三级缓存来解决循环依赖问题。三级缓存包括:

  1. 一级缓存(singletonObjects):存放完全初始化好的单例Bean。
  2. 二级缓存(earlySingletonObjects):存放提前暴露的单例Bean,通常是原始Bean对象,用于解决循环依赖。
  3. 三级缓存(singletonFactories):存放能够生成Bean的Factory,用于生成Bean的代理对象。

解决循环依赖的步骤

  1. 实例化Bean:Spring首先通过反射机制实例化Bean对象,但此时Bean还没有进行属性注入。
  2. 将Bean添加到三级缓存:将实例化但未初始化的Bean包装成ObjectFactory,放入三级缓存singletonFactories中。
  3. 属性注入:Spring会尝试进行属性注入,如果遇到循环依赖,则会从三级缓存中获取依赖的Bean。
  4. 提前暴露Bean:如果发现三级缓存中存在依赖的Bean,则通过ObjectFactory从三级缓存中获取Bean,并将其放入二级缓存earlySingletonObjects中。
  5. 初始化Bean:完成属性注入后,Spring会初始化Bean(例如,调用InitializingBean接口的afterPropertiesSet方法)。
  6. 将Bean移至一级缓存:初始化完成后,将Bean从二级缓存中移到一级缓存singletonObjects中。

示例

以下是一个简单的示例代码,展示了两个相互依赖的Bean及其解决方式。

示例代码
java">import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Component
public class A {private B b;@Autowiredpublic void setB(B b) {this.b = b;}public void doSomething() {System.out.println("A is doing something");}
}@Component
public class B {private A a;@Autowiredpublic void setA(A a) {this.a = a;}public void doSomething() {System.out.println("B is doing something");}
}import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;@Configuration
@ComponentScan(basePackages = "com.example")
public class AppConfig {public static void main(String[] args) {ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);A a = context.getBean(A.class);a.doSomething();B b = context.getBean(B.class);b.doSomething();}
}

在上述代码中,AB相互依赖。当Spring容器启动时,Spring会按照前述的三级缓存机制解决这两个Bean之间的循环依赖问题。

总结

Spring通过三级缓存机制(一级缓存、二级缓存和三级缓存)解决循环依赖问题。这种机制允许Spring在Bean尚未完全初始化之前,提前暴露Bean引用,从而打破循环依赖的环。这种设计既保证了Bean的依赖注入,又避免了循环依赖导致的死锁问题。


http://www.ppmy.cn/ops/89691.html

相关文章

【微服务】Spring Cloud 服务网关之Zuul

文章目录 强烈推荐引言用途使用情况使用场景基本使用示例总结强烈推荐专栏集锦写在最后 强烈推荐 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站:人工智能 引言 服务网关(AP…

MYSQL 事务 与 索引

文章目录 MySQL事务MySQL的自动提交模式使用事务1 隔离级别1 脏读2 幻读3 不可重复读 3 MySQL索引1 创建索引2 删除索引 参考 MySQL事务 事务是一组SQL语句的执行,被视为一个单独的工作单元 事务必须满足以下4个条件:ACID 原子性(Atomicit…

山东大学考研机试题——整数序列

题目描述 传送门——AcWing 3717. 整数序列 - AcWing 很多整数可以由一段连续的正整数序列(至少两个数)相加而成,比如 2534567121325345671213。 输入一个整数 N,输出 N 的全部正整数序列,如果没有则输出 NONE。 输…

VS Code 和 Visual Studio 哪个更好

文章目录 VS Code 和 Visual Studio 哪个更好Visual Studio Code简介Visual Studio简介相同点差异点总结 VS Code 和 Visual Studio 哪个更好 Visual Studio Code简介 Visual Studio Code(简称 VS Code)是一款开源的、免费的、跨平台的、轻量级的代码编…

新型蜜罐有哪些?未来方向如何?

前言:技术发展为时代带来变革,同时技术创新性对蜜罐产生推动力。 一、新型蜜罐的诞生 技术发展为时代带来变革,同时技术创新性对蜜罐产生推动力,通过借鉴不同技术思想、方法,与其它技术结合形成优势互补,…

webpack的loader机制

webpack的loader机制 loader本质上就是导出函数的JavaScript模块。导出的函数,可以用来实现内容的转换。 /* * param{string|Buffer} content 源文件的内容 * param{object} [map] SourceMap数据 * param{any} [meta] meta数据,可以是任何数据 * */ fu…

数据湖和数据仓库核心概念与对比

随着近几年数据湖概念的兴起,业界对于数据仓库和数据湖的对比甚至争论就一直不断。有人说数据湖是下一代大数据平台,各大云厂商也在纷纷的提出自己的数据湖解决方案,一些云数仓产品也增加了和数据湖联动的特性。但是数据仓库和数据湖的区别到…

DC-6靶机渗透测试

DC-6靶机 文章目录 DC-6靶机信息收集web渗透权限获取权限获取方式1 -- 利用45274.html权限获取方式2 -- 利用50110.py权限提升 信息收集 通过对靶机进行扫描,得出IP地址为192.168.78.134,开放端口为80 和 22 ssh,肯定会进行ssh连接的了 访问…