【CVTE 一面凉经Ⅰ】循环依赖如何解决

news2025/1/12 9:05:29

目录

  • 一.🦁 开始前的废话
  • 二. 🦁 什么是循环依赖?
  • 三. 🦁Spring 容器解决循环依赖的原理是什么?
  • 五. 🦁 三级缓存解决循环依赖的原理
  • 六. 🦁 由有参构造方法注入属性的循环依赖如何解决?
  • 七.🦁 ENDing

权限管理

一.🦁 开始前的废话

最近一个小伙伴"李四"面 C 厂后台开发凉凉了,原因是面试官就他简历的微服务项目问了个很常见的面试题,他没答上来!现在咱们来看看面试官和李华的对话:

面试官问道:“你的项目如果遇到循环依赖了咋办?”

李四见状惊喜万分,笑着回答,确实遇到过,官方推荐我在 yaml 文件添加这段代码来解决:

spring:
  main:
    allow-circular-references: true

面试官:“嗯呢,这个可以!这段代码什么意思?那如果是通过构造函数导入的实例,还能使用这段代码解决嘛?”

李四:我 😥…

面试官:“好的,没关系。回家等消息叭!”


经过面试官二连问,李华败下阵来。那我们现在来讨论一下这个循环依赖应该怎么解决!从源头开始剖析。

二. 🦁 什么是循环依赖?

其实循环依赖是指在我们 Coding 的过程中,由于不好的设计,导致两个或以上的 Bean 相互依赖导致形成了一个闭环(lion 依赖 tiger,反过来 tiger 也在依赖 lion)。在 Springboot 2.6 以前,spring 容器是可以在实例化 Bean 的过程中,是可以自动解决一部分循环依赖的问题(依靠三级缓存),由于这个方案导致了越来越多的 Coder 老是滥用,导致代码质量越来越差,所以从 SpringBoot 2.6 起,就默认把这种方案给禁用了。如果你的项目里还存在循环依赖,SpringBoot 将拒绝启动!

***************************
APPLICATION FAILED TO START
***************************
Description:
The dependencies of some of the beans in the application context form a cycle:
┌─────┐
|  tiger defined in file [/Users/study/personal-projects/easy-web/target/classes/com/log/web/controller/Lion.class]
↑     ↓
|  lion (field private com.log.web.controller.Tiger com.log.web.service.Tiger.class)
└─────┘

需要我们在 yaml 或者 properties 文件配置参数来临时开启循环依赖(开启方式就是上面那段代码)。

三. 🦁Spring 容器解决循环依赖的原理是什么?

前面我们说了 Spring 容器可以处理部分循环依赖问题,只不过是高版本的 Spring 需要手动开启。那么 Spring 是如何解决这个问题的呢?

其实 Spring 是依靠三级缓存的方式来处理循环依赖的

那么它是如何检测存在循环依赖的呢?

其实也比较简单,就是容器在创建 实例A 的过程中,会给它贴一个正在创建的标签,说明它正在创建了,然后递归去创建其依赖的 实例B,当 实例B 也依赖 A 时,并且发现其正处于创建中的状态,那么就说明存在循环依赖了。

那么三级缓存是如何来解决它呢?

首先,我们可以认为实例化一个对象可以简单分为两步:为这个对象填充所需要的属性,而填充对象的属性的方式又有两种:
Spring 容器解决循环依赖可以理解为是在一个闭环中,先将第一个实例A实例化并提前暴露出来,这样闭环上依赖A的B就可以创建完成,那么依此类推,依赖 B 的 C 也可以创建出来了,从而递归可以顺利进行下去,当跳出最后一层递归后,A依赖的D也创建出来了,再将D注入到A上,整个闭环的实例就完成创建了。

  • 实例化该对象

  • 为这个对象填充所需要的属性,而填充对象的属性的方式又有两种:

    • 有参构造方法直接在对象创建时填充进去(Spring 无法自动处理这种方式创建实例的循环依赖);
    • 通过 set() 方法填充对象属性,而三级缓存仅对这种方法有效。

Spring 容器解决循环依赖可以理解为是在一个闭环中,先将第一个实例A实例化并提前暴露出来,这样闭环上依赖A的B就可以创建完成,那么依此类推,依赖 B 的 C 也可以创建出来了,从而递归可以顺利进行下去,当跳出最后一层递归后,A依赖的D也创建出来了,再将D注入到A上,整个闭环的实例就完成创建了

那么为什么说只有对于 set() 方法填充对象属性的方式,Spring容器才能解决呢?原因很简单(以B依赖A,A依赖B为例),A类 通过有参构造的方式在创建实例并同时将属性注入,那么在这个过程中(注意:此时A并没有完成实例创建),它会去寻找它依赖的对象B,此时B类也开始创建实例,但是由于A依赖B,并且A类并没有完成实例创建,所以二者就处在这种尴尬状态,导致最后容器报错!

然而,若是通过 set() 方法填充对象属性的方式,那么此时实例A已经创建完成,只是还没有注入对应的属性(这个我们后文暂且叫装配Bean吧,因为所谓的实例本质上就是一个Bean对象)而已。这就是为什么 Spring 容器为什么只能解决 set() 方法装配 Bean对象的循环依赖。

五. 🦁 三级缓存解决循环依赖的原理

我们前面提了好多次三级缓存!那么三级缓存到底是什么?它是如何解决循环依赖的?

Spring 容器的三个缓存分别如下:

// 1级缓存:存放实例化+属性注入+初始化+代理(如果有代理)后的单例bean
    private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);
    // 2级缓存:存放实例化+代理(如果有代理)后的单例bean
    private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);
    // 3级缓存:存放封装了单例bean(实例化的)的对象工厂
    private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);

我们都知道 Spring 容器通过 IOC 创建单例对象,这个对象创建完成后最终都是存放在 singletonObjects 一级缓存里面的,但并不是所有的实例一开始创建就存进一级缓存!创建的时候需要放进不同的缓存。(具体的不讨论,我们现在来看一下循环依赖过程中,三级缓存是如何工作的!)

以 实例A 与 实例B 相互依赖为例子:

首先 Spring 容器先创建 A实例,实例化完成后将其封装为 ObjectFactory 对象先存入到三级缓存 singletonFactories 中;然后容器对 A 做进一步的装配,装配的时候发现 A 依赖 B ,所以 Spring 去三级缓存中寻找 B,发现其还没有创建,所以会先创建 B实例,B 创建完成后被封装为 ObjectFactory 对象被存入三级缓存 singletonObjects ,同时也会先进行装配,其间 Spring 也发现了 B 依赖于 A ,所以会回到三级缓存中寻找 A实例,终于在第三级缓存中发现了被封装为 ObjectFactory 对象的 A,将其取出来通过 getObject() 方法得到 A,拿到 A 后不再将其放回三级缓存,而是存进二级缓存 earlySingletonObjects 中,而三级缓存中的 ObjectFactoryA 也会被移除,这个过程相当于是 A 从三级缓存——>二级缓存。同时也将 A 填充给 B,至此B 完成装配,从三级缓存——>二级缓存,Spring 容器不会忘记还在"嗷嗷待哺"的 实例A,回过头去一级缓存找到 B将其填充给 A,至此 A 完成装配,从二级缓存——>一级缓存,创建结束,循环依赖完美解决。

六. 🦁 由有参构造方法注入属性的循环依赖如何解决?

我们通过一个案例来说明:实现两个相互依赖的类 A B:


@Component
public class A{

    private B b;

    public A(B b) {
        this.b = b;
    }
}

@Component
public class B {
    private A a;

    public B(A a) {
        this.a = a;
    }
}

启动容器就会发现如下报错:

The dependencies of some of the beans in the application context form a cycle:

┌─────┐
|  b defined in file [.../target/classes/com/demo/service/B.class]
↑     ↓
|  a defined in file [.../target/classes/com/demo/service/A.class]
└─────┘

我们通过在 A/B 类上的构造函数添加 @Lazy 注解则会解决这个循环依赖问题。如下:

@Component
public class A{
    private B b;

    public A(@Lazy B b) {
        this.b = b;
    }
}

此时则会正常启动了!!!

那么这其中的原理是什么呢?请看下回分解!

七.🦁 ENDing

错过的题目,自己要学会成长,下次再也不错啦!


在这里插入图片描述

🦁 其它优质专栏推荐 🦁

🌟《Java核心系列(修炼内功,无上心法)》: 主要是JDK源码的核心讲解,几乎每篇文章都过万字,让你详细掌握每一个知识点!

🌟 《springBoot 源码剥析核心系列》:一些场景的Springboot源码剥析以及常用Springboot相关知识点解读

欢迎加入狮子的社区:『Lion-编程进阶之路』,日常收录优质好文

更多文章可持续关注上方🦁的博客,2023咱们顶峰相见!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1536994.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【算法分析与设计】相同的树

题目 给你两棵二叉树的根节点 p 和 q &#xff0c;编写一个函数来检验这两棵树是否相同。 如果两个树在结构上相同&#xff0c;并且节点具有相同的值&#xff0c;则认为它们是相同的。 示例 示例 1&#xff1a; 输入&#xff1a;p [1,2,3], q [1,2,3] 输出&#xff1a;true示…

IDEA创建Maven项目实现数据库表查询

在此之前&#xff0c;务必确保你本地环境已经正确配置了 JDK 和 Maven&#xff0c;并且相关的环境变量已经设置。这是你踏上编程之旅的第一步&#xff0c;也是你与代码交互的基础。可通过命令提示符中输入 java -version 和 mvn -v 命令&#xff0c;以验证你的 JDK 和 Maven 是…

多ip多进程代理的实现方法

目录 写在前面 一、背景 二、实现方法 1. 使用多线程处理代理请求 2. 使用多进程处理代理请求 3. 实现多IP代理 三、总结 写在前面 实现多IP多进程代理需要使用Python的多线程和多进程模块。本文将介绍如何使用这些模块来实现多IP多进程代理&#xff0c;并提供相关的代…

蓝桥杯练习——神秘咒语——axios

目标 完善 index.js 中的 TODO 部分&#xff0c;通过新增或者修改代码&#xff0c;完成以下目标&#xff1a; 点击钥匙 1 和钥匙 2 按钮时会通过 axios 发送请求&#xff0c;在发送请求时需要在请求头中添加 Authorization 字段携带 token&#xff0c;token 的值为 2b58f9a8-…

五.java数组

1.int [] arr{1,2,3} 2.lenarr.length; 3.数组内存执行原理 java程序编译后产生一个class文件然后提交到内存中的虚拟机JVM中去运行&#xff0c;java为了便于虚拟机运行程序就将虚拟机的这块内存划分为5个区域&#xff1a;方法区&#xff0c;栈&#xff0c;堆。。。 经典函数…

Matlab快捷键与函数

注释&#xff1a;注释对于代码的重要性我们就不做过多的解释了。不做注释的代码不是好代码。选中要注释的语句&#xff0c;按快捷键CtrlR,或者在命令行窗口上面的注释地方可以进行注释。当然也可以直接在语句前面“%”就可以&#xff08;注意&#xff1a;一定要用英文符号&…

BigDecimal类的使用,用于精确计算任意精度的数字

BigDecimal类 BigDecimal 是 Java 中用于精确表示任意精度的十进制数的类。在很多情况下,使用基本数据类型(如 double 或 float)进行浮点数计算可能会导致精度丢失或舍入错误。BigDecimal 提供了一种更精确的解决方案,可以处理需要高精度计算的场景,比如财务应用或科学计算…

29-3 哥斯拉安装使用

环境准备:构建完善的安全渗透测试环境:推荐工具、资源和下载链接_渗透测试靶机下载-CSDN博客 一、哥斯拉 (Godzilla) 介绍 哥斯拉是一个基于流量、HTTP全加密的webshell管理工具,具有以下特点: 内置了3种Payload以及6种加密器,6种支持脚本后缀,20个内置插件基于Java,可…

第十三届蓝桥杯省赛真题 Java B 组【原卷】

文章目录 发现宝藏【考生须知】试题 A: 星期计算试题 B: 山试题 C: 字符统计试题 D: 最少刷题数试题 E \mathrm{E} E : 求阶乘试题 F : \mathrm{F}: F: 最大子矩阵试题 G: 数组切分试题 H: 回忆迷宫试题 I: 红绿灯试题 J 拉箱子 发现宝藏 前些天发现了一个巨牛的人工智能学习…

仿muduo库实现one thread one loop式并发服务器

文章目录 一、项目简介 二、项目整体认识 2、1 HTTP服务器 2、2 Reactor模型 三、预备知识 3、1 C11 中的 bind 3、2 简单的秒级定时任务实现 3、3 正则库的简单使用 3、4 通用类型any类型的实现 四、服务器功能模块划分与实现 4、1 Buffer模块 4、2 Socket模块 4、3 Channel模…

【TD3思路及代码】【自用笔记】

1 组成&#xff08;Target Network Delayed Training&#xff09; Actor网络&#xff1a;这个网络负责根据当前的状态输出动作值。在训练过程中&#xff0c;Actor网络会不断地学习和优化&#xff0c;以输出更合适的动作。Critic网络&#xff1a;TD3中有两个Critic网络&#xff…

解决:springboot项目访问hdfs文件提示guava版本不兼容

1、问题描述 版本说明&#xff1a;我用的hadoop版本&#xff1a;3.1.3 项目可以正常启动&#xff0c;但是调用访问hdfs的服务时候报错,报错消息如下&#xff1a;com.google.common.base.preconditions.checkArgument(ZL java/lang/String;Ljava/lang/Object:)V 原因分析&#x…

力扣 字符串解码

维护一个放数字的栈&#xff0c;一个放字母的栈 遇到[把数字和字母入栈&#xff0c;遇到]把当前字母循环加上数字栈头遍的字母栈头 class Solution { public:string decodeString(string s) {string ans"";stack<int>sz;stack<string>zm;里面是string …

蓝桥杯(3.22 刷真题)

P8682 [蓝桥杯 2019 省 B] 等差数列 RE是因为除以0的情况 import java.util.Arrays; import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner sc new Scanner(System.in);int n sc.nextInt();int[] res new int[n1];for(int i1;i&l…

集合深入------理解底层。

集合的使用 前提&#xff1a;栈、堆、二叉树、hashcode、toString()、quesalus()的知识深入和底层理解。 1、什么是集合 集合就是咋们所说的容器 ​ 前面我们学习过数组 数组也是容器 ​ 容器&#xff1a;装东西的 生活中有多少的容器呀? 水杯 教室 酒瓶 水库 只要是…

【论文速读】| 视觉对抗样本:突破对齐的大语言模型

本次分享论文为&#xff1a;Visual Adversarial Examples: Jailbreak Aligned Large Language Models 基本信息 原文作者&#xff1a;Xiangyu Qi, Peter Henderson, Kaixuan Huang, Ashwinee Panda, Mengdi Wang, Prateek Mittal 作者单位&#xff1a;普林斯顿大学、斯坦福大…

算法打卡day20|二叉树篇09|Leetcode 669. 修剪二叉搜索树、108.将有序数组转换为二叉搜索树、538.把二叉搜索树转换为累加树

算法题 Leetcode 669. 修剪二叉搜索树 题目链接:669. 修剪二叉搜索树 大佬视频讲解&#xff1a;修剪二叉搜索树视频讲解 个人思路 把这道题想复杂了&#xff0c;还想着如何去重构树 解法 递归法 依旧递归三步走 1.确定递归函数的参数以及返回值 这题递归需要返回值&#…

NVMe开发——NAND Flash的基本原理

1. 存储单元 1.1. 半导体 半导体&#xff08;semiconductor&#xff09;指常温下导电性能介于导体与绝缘体之间的材料。纯净半导体一般是四价原子(如硅或锗)。 1.1.1. P型半导体 P型半导体是通过将一个纯净的半导体材料&#xff08;如硅或锗&#xff09;中掺杂少量的三价杂…

软件管理rpm与yum

源代码包下载 Compare, Download & Develop Open Source & Business Software - SourceForgehttps://sourceforge.net/ rpm包下载 Welcome to the RPM repository on fr2.rpmfind.nethttp://rpmfind.net/linux/RPM/ 软件包管理 1.rpm包管理: 1)查询: 安装…

Linux网络协议栈从应用层到内核层②

文章目录 1、bind 源码剖析2、listen 源码剖析3、accept 源码剖析4、connect 源码剖析客户端调用connect成功&#xff0c;但三次握手并未完成&#xff0c;进程是如何阻塞自己客户端在connect时&#xff0c;如何选择源端口客户发送syn封包以及重传服务端收到syn封包&#xff0c;…