循环依赖通常发生在两个或多个Spring Bean之间,它们通过构造器、字段(使用@Autowired)或setter方法相互依赖,从而形成一个闭环。下面是一个使用字段注入(即使用@Autowired)导致的循环依赖的示例:

示例代码:

假设我们有两个类,ClassA 和 ClassB,它们相互依赖:

public class ClassA {  

    @Autowired
private ClassB classB; // ... 其他代码 ...
} @Component
public class ClassB { @Autowired
private ClassA classA; // ... 其他代码 ...
}

在上面的示例中,ClassA 依赖 ClassB,而 ClassB 又依赖 ClassA。当Spring容器启动时,它会尝试为这两个类创建bean的实例。但是,由于它们之间的循环依赖,这会导致问题。

问题说明:

当Spring容器创建ClassA的bean时,它会发现ClassA依赖于ClassB,所以它会尝试创建ClassB的bean。当Spring容器创建ClassB的bean时,它又会发现ClassB依赖于ClassA,但此时ClassA的bean还没有被完全初始化(因为它正在等待ClassB的bean),这就形成了一个死循环。

那么Spring是如何解决这种循环依赖问题的?

三级缓存机制:

Spring容器在创建bean的过程中,会维护三个缓存,分别是:singletonObjects(一级缓存)、earlySingletonObjects(二级缓存)和singletonFactories(三级缓存)。

当Spring容器开始实例化一个bean时,会先将其ObjectFactory(对象工厂)放入三级缓存中。如果在实例化过程中需要注入其他bean,并且这个bean也正在实例化中,Spring会先从一级缓存中查找该bean的实例。如果没有找到,会到二级缓存中查找。如果二级缓存中也没有,那么会到三级缓存中查找ObjectFactory,并调用ObjectFactory的getObject方法来获取bean的实例。

当获取到bean的实例后,会将其放入二级缓存中,并从三级缓存中移除ObjectFactory。

当bean的实例化过程完成后,会将其放入一级缓存中。

通过这种方式,Spring可以在bean的实例化过程中解决循环依赖问题。

@Lazy注解:

在Spring中,可以使用@Lazy注解来延迟bean的初始化。当一个bean被标记为@Lazy时,Spring容器在启动时不会立即实例化它,而是在第一次被使用时才进行实例化。

通过将循环依赖的bean声明为懒加载,可以延迟它们的初始化过程,从而避免在容器启动时发生循环依赖问题。

需要注意的是,@Lazy注解只能用于单例作用域的bean,并且要求依赖项必须是接口类型。

以下是使用@Lazy注解来解决循环依赖的示例代码:

@Component
public class ClassA { private final ClassB classB; // 使用@Autowired和@Lazy注解来延迟ClassB的注入
@Autowired
public ClassA(@Lazy ClassB classB) {
this.classB = classB;
} // ... 其他代码 ...
} @Component
public class ClassB { private final ClassA classA; // 使用@Autowired和@Lazy注解来延迟ClassA的注入
@Autowired
public ClassB(@Lazy ClassA classA) {
this.classA = classA;
} // ... 其他代码 ...
}

在上面的示例中,@Lazy注解被用于ClassA和ClassB的构造器参数上,以延迟它们之间的依赖注入。这意味着在创建ClassA时,它不会立即尝试去初始化ClassB,而是会得到一个代理对象。同样,在创建ClassB时,它也不会立即初始化ClassA。

 

避免构造器循环依赖:

在Spring中,构造器循环依赖是无法解决的,因为构造器在实例化bean的过程中被调用,如果两个bean相互依赖对方的构造器,那么就会形成死锁。

因此,在设计bean的依赖关系时,应该尽量避免使用构造器注入来创建循环依赖。可以使用setter注入或字段注入来代替构造器注入。

下面是一个构造器循环依赖的错误代码示例:

@Component
public class ClassA { private final ClassB classB; // 通过构造器注入ClassB,形成循环依赖
@Autowired
public ClassA(ClassB classB) {
this.classB = classB;
} // ... 其他代码 ...
} @Component
public class ClassB { private final ClassA classA; // 通过构造器注入ClassA,与ClassA形成循环依赖
@Autowired
public ClassB(ClassA classA) {
this.classA = classA;
} // ... 其他代码 ...
}

在上面的示例中,ClassA和ClassB各自在构造函数中依赖于对方,这就形成了构造器循环依赖。

当Spring容器尝试创建这两个bean的实例时,会遇到问题:

  • Spring首先尝试创建ClassA的实例,并发现它需要ClassB的实例。

  • 然后,Spring尝试创建ClassB的实例,但发现它需要ClassA的实例。

  • 由于ClassA的实例正在等待ClassB的实例,而ClassB的实例又正在等待ClassA的实例,这导致了一个死循环,无法继续创建bean的实例。

综上所述,Spring通过三级缓存机制、@Lazy注解以及避免构造器循环依赖等方式来解决循环依赖问题。这些机制使得Spring容器能够更加灵活地处理bean之间的依赖关系,提高系统的可维护性和可扩展性。

Java面试题:Spring中的循环依赖,给程序员带来的心理阴影的更多相关文章

  1. 面试必杀技,讲一讲Spring中的循环依赖

    本系列文章: 听说你还没学Spring就被源码编译劝退了?30+张图带你玩转Spring编译 读源码,我们可以从第一行读起 你知道Spring是怎么解析配置类的吗? 配置类为什么要添加@Configu ...

  2. 面试阿里,腾讯,字节跳动90%都会被问到的Spring中的循环依赖

    前言 Spring中的循环依赖一直是Spring中一个很重要的话题,一方面是因为源码中为了解决循环依赖做了很多处理,另外一方面是因为面试的时候,如果问到Spring中比较高阶的问题,那么循环依赖必定逃 ...

  3. 从一部电影史上的趣事了解 Spring 中的循环依赖问题

    title: 从一部电影史上的趣事了解 Spring 中的循环依赖问题 date: 2021-03-10 updated: 2021-03-10 categories: Spring tags: Sp ...

  4. 【Spring】Spring中的循环依赖及解决

    什么是循环依赖? 就是A对象依赖了B对象,B对象依赖了A对象. 比如: // A依赖了B class A{ public B b; } // B依赖了A class B{ public A a; } ...

  5. Spring中的循环依赖解决详解

    前言 说起Spring中循环依赖的解决办法,相信很多园友们都或多或少的知道一些,但当真的要详细说明的时候,可能又没法一下将它讲清楚.本文就试着尽自己所能,对此做出一个较详细的解读.另,需注意一点,下文 ...

  6. Spring中解决循环依赖报错的问题

    什么是循环依赖 当一个ClassA依赖于ClassB,然后ClassB又反过来依赖ClassA,这就形成了一个循环依赖: ClassA -> ClassB -> ClassA 原创声明 本 ...

  7. 一起来踩踩 Spring 中这个循环依赖的坑

    1. 前言 2. 典型场景 3. 什么是依赖 4. 什么是依赖调解 5. 为什么要依赖注入 6. Spring的依赖注入模型 7. 非典型问题 参考资料 1. 前言 这两天工作遇到了一个挺有意思的Sp ...

  8. Spring中的循环依赖

    循环依赖 在使用Spring时,如果主要采用基于构造器的依赖注入方式,则可能会遇到循环依赖的情况,简而言之就是Bean A的构造器依赖于Bean B,Bean B的构造器又依赖于Bean A.在这种情 ...

  9. Spring 如何解决循环依赖问题?

    在关于Spring的面试中,我们经常会被问到一个问题,就是Spring是如何解决循环依赖的问题的. 这个问题算是关于Spring的一个高频面试题,因为如果不刻意研读,相信即使读过源码,面试者也不一定能 ...

  10. Spring如何解决循环依赖问题

    目录 1. 什么是循环依赖? 2. 怎么检测是否存在循环依赖 3. Spring怎么解决循环依赖 本文主要是分析Spring bean的循环依赖,以及Spring的解决方式. 通过这种解决方式,我们可 ...

随机推荐

  1. Promise + Async&Await + Array.reduce + 函数递归 解决网络/接口请求的依次/排队不间断间隔访问

    背景 试想在一个需要频繁更新数据的场景(例如:监控.图表类),常规方法是设置一个间隔 N 秒的定时器 setInterval:但是这种方式存在一个问题,当前一个请求时间过长时(超过了间隔时间),后一个 ...

  2. Jetty的ssl模块

    启用ssl模块,执行如下命令: java -jar $JETTY_HOME/start.jar --add-modules=ssl 命令的输出,如下: INFO : ssl initialized i ...

  3. JDK11的新特性:新的HTTP API

    目录 简介 使用HTTP Client请求的基本流程 创建HTTP Client 创建HTTP Request 发送HTTP请求 异步HTTP请求 总结 JDK11的新特性:新的HTTP API 简介 ...

  4. Mybatis实现增删改查

    ​1.CRUD 1.1namespace namespace中的包名必须和Dao/mapper接口包名一致 1.2select 选择,查询语句 id:就是对应的namespace中的方法名 resul ...

  5. Linux之openssl实现私有CA

    一.简介 Centos7.9通过openssl工具构建一个私有的CA,用于颁发证书. 验证私有CA为httpd应用签署证书 二.构建私有CA 1.编辑CA的配置文件 [root@HLWHOST tls ...

  6. 动态规划(三)——线性dp

    一.概念 具有线性阶段划分的动态规划算法叫作线性动态规划(简称线性DP).若状态包含多个维度,则每个维度都是线性划分的阶段,也属于线性DP,如下图所示: 二.线性dp的三大经典例题 1.LIS问题:求 ...

  7. 黑客终端qsnctfwp

    进入网页,发现是网页版的 cmd (/doge) 输入ls发现输出了以下内容 按 F12 检查代码,在<script>中发现输入命令为cat /flag则可获得 flag 此时即可直接复制 ...

  8. ASP.NET MVC 性能优化和调试

    学习 .NET Core 应用程序的调试技术可以分为以下步骤: 理解基础概念:首先,你需要理解什么是调试以及为什么我们需要调试.理解断点.单步执行.变量监视等基本调试概念. 学习 Visual Stu ...

  9. vue+scss混合(mixins)使用(css代码的vuex(公共管理))

    scss混合(mixins)使用 例一.使用混合mixins中的变量来定义一个n行文本溢出隐藏的公用样式. 1.创建mixins.scss文件 //文本n行溢出隐藏 @mixin ellipsisBa ...

  10. JVM简明笔记3:类加载机制

    1 类的加载 类的加载指的是将类的 .class 文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个 java.lang.Class 对象,用来封装类在方法区内的数据结 ...