Spring如何通过三级缓存解决循环依赖问题?

news2024/12/23 22:08:56

目录

一、什么是Spring

二、循环依赖问题

三、三级缓存机制

四、如何通过三级缓存解决循环依赖问题


 

一、什么是Spring

Spring框架是一个开源的Java应用程序开发框架,提供了一种全面的、一致的编程模型,用于构建企业级应用程序和服务。它由Rod Johnson在2003年创建,旨在简化Java开发并促进松耦合、可维护性和可扩展性。

Spring框架的核心特性包括: 1.控制反转(IoC):通过IoC容器管理对象之间的依赖关系,降低了组件之间的耦合度,使得应用程序更加灵活和可测试。 2.面向切面编程(AOP):通过AOP模块,可以将横切关注点(如日志记录、事务管理等)从业务逻辑中分离出来,使得代码更加模块化和可维护。 3.数据访问和集成:Spring提供了灵活的数据访问和集成支持,包括对关系型数据库、NoSQL数据库、消息队列、缓存等的支持。 4.Web开发:Spring框架提供了Spring MVC模块,用于构建Web应用程序。它基于MVC设计模式,提供了灵活的配置和处理请求的能力。 5.事务管理:Spring框架支持声明式事务管理,通过配置来管理事务的边界和传播规则,简化了事务管理的编程工作。 6.安全性:Spring框架提供了一套安全性框架,用于认证和授权管理,保护应用程序的安全性。 7.测试支持:Spring框架提供了广泛的测试支持,包括对单元测试、集成测试和端到端测试的支持,使得开发人员能够更轻松地编写和运行测试用例。

Spring框架的设计理念是轻量级、可扩展和可插拔的,它被广泛应用于Java企业级开发中,是目前最受欢迎的Java开发框架之一。

 

二、循环依赖问题

Spring的循环依赖问题指的是在使用Spring的IoC容器进行对象创建和依赖注入时,如果存在循环依赖关系,则可能导致创建对象的过程无法完成或出现错误。

循环依赖是指两个或多个Bean之间相互依赖,形成一个循环链条。当A依赖B,B又依赖A时,就会产生循环依赖。

Spring在处理循环依赖时使用了三级缓存(singletonFactories、earlySingletonObjects和singletonObjects)来解决问题。具体的解决过程如下:

  1. 首先,当创建一个Bean时,Spring将该Bean放入singletonFactories缓存中。

  2. 如果Bean的创建过程中需要依赖其他Bean,Spring会通过递归调用创建所需的其他Bean。

  3. 当创建另一个Bean时,如果发现该Bean已经在singletonFactories缓存中,说明发生了循环依赖。

  4. 在这种情况下,Spring会尝试从earlySingletonObjects缓存中获取Bean的早期实例,如果存在,则返回该实例,否则继续创建Bean。

  5. 如果在创建Bean的过程中依然无法解决循环依赖问题,Spring会抛出BeanCurrentlyInCreationException异常,表示无法完成Bean的创建。

为了避免循环依赖问题,可以考虑以下几种方式:

  1. 通过构造函数注入:使用构造函数注入依赖,而不是使用Setter方法注入依赖。

  2. 使用Lazy注解:使用@Lazy注解延迟初始化Bean,以避免过早创建循环依赖的Bean。

  3. 使用Setter方法注入:将依赖注入改为Setter方法注入,并使用@Autowired注解。

  4. 使用@PostConstruct注解:使用@PostConstruct注解在Bean创建完成后执行一些初始化操作。

需要注意的是,虽然Spring提供了解决循环依赖的机制,但是过多的循环依赖可能会导致性能下降和代码的可读性下降,因此在设计应用程序时,应尽量避免出现循环依赖的情况。

 

三、三级缓存机制

Spring的三级缓存机制是为了解决循环依赖问题而设计的。当使用Spring的IoC容器创建对象时,会经过三个缓存级别来解决循环依赖问题。

  1. singletonFactories缓存:在对象创建过程中,如果发现循环依赖,Spring会将正在创建的Bean放入singletonFactories缓存中。这个缓存中存放的是对象的提供者,也就是用来创建Bean的工厂。

  2. earlySingletonObjects缓存:如果在创建Bean的过程中,发现依赖的Bean已经在singletonFactories缓存中,说明发生了循环依赖。此时,Spring会尝试从earlySingletonObjects缓存中获取Bean的早期实例(还未完全初始化),以解决循环依赖问题。

  3. singletonObjects缓存:如果无法从earlySingletonObjects缓存中获取到早期实例,Spring会将Bean放入singletonObjects缓存中,用于存放完全初始化后的Bean。

三级缓存的工作流程如下:

  1. 当创建Bean时,首先会检查singletonFactories缓存,如果发现正在创建的Bean已经在缓存中,则说明发生了循环依赖。

  2. 在循环依赖的情况下,会尝试从earlySingletonObjects缓存中获取早期实例,如果成功获取到早期实例,则返回该实例,解决了循环依赖问题。

  3. 如果无法从earlySingletonObjects缓存中获取早期实例,表示无法解决循环依赖,Spring会抛出BeanCurrentlyInCreationException异常。

  4. 如果解决了循环依赖,Bean会继续完成创建过程,并最终放入singletonObjects缓存中,供其他Bean使用。

通过三级缓存机制,Spring能够解决大部分的循环依赖问题,并确保Bean的创建和注入顺利进行。但需要注意的是,过多的循环依赖会增加系统的复杂性和性能开销,因此在设计应用程序时,应尽量避免循环依赖的出现。

以下是一个简单的Java样例代码,演示了Spring的三级缓存机制的实现:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;
​
@Component
@Scope("singleton")
public class BeanA {
​
    private BeanB beanB;
​
    @Autowired
    public BeanA(BeanB beanB) {
        this.beanB = beanB;
    }
​
    public void doSomething() {
        System.out.println("BeanA is doing something");
        beanB.doSomething();
    }
}
​
@Component
@Scope("singleton")
public class BeanB {
​
    private BeanA beanA;
​
    @Autowired
    public BeanB(BeanA beanA) {
        this.beanA = beanA;
    }
​
    public void doSomething() {
        System.out.println("BeanB is doing something");
        beanA.doSomething();
    }
}
​
public class MainApp {
​
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(Config.class);
        BeanA beanA = context.getBean(BeanA.class);
        beanA.doSomething();
    }
}

 

在上面的代码中,BeanABeanB互相依赖。当BeanA创建时,它依赖于BeanB。当BeanB创建时,它又依赖于BeanA。在MainApp中,通过ApplicationContext获取BeanA的实例,并调用doSomething方法。Spring会自动处理循环依赖关系,通过三级缓存机制确保BeanABeanB成功创建和注入。

需要注意的是,上述代码中使用了注解驱动的配置方式,通过@Component注解将Bean注册到Spring容器中,使用@Autowired注解进行依赖注入。此外,可以通过@Scope注解指定Bean的作用域为singleton,这是默认的作用域,也是三级缓存机制生效的前提。

四、如何通过三级缓存解决循环依赖问题

Spring通过三级缓存来解决循环依赖问题的具体步骤如下:

  1. 创建Bean A,将该Bean放入singletonFactories缓存中。

  2. 当创建Bean A的过程中发现它需要依赖Bean B,Spring会先检查singletonObjects缓存中是否存在Bean B的实例。

  3. 如果singletonObjects缓存中存在Bean B的实例,说明Bean B已经创建完成,可以直接将其注入到Bean A中。

  4. 如果singletonObjects缓存中不存在Bean B的实例,Spring会检查singletonFactories缓存中是否存在Bean B的提供者(即用来创建Bean B的工厂)。

  5. 如果singletonFactories缓存中存在Bean B的提供者,说明Bean B正在创建过程中,但尚未创建完成。此时,Spring会将正在创建的Bean A放入earlySingletonObjects缓存中,表示Bean A的早期实例。

  6. Spring继续创建Bean B,当Bean B创建完成后,会将其放入singletonObjects缓存中,并从earlySingletonObjects缓存中获取Bean A的早期实例。

  7. Bean A和Bean B创建完成后,Spring会完成Bean A的注入操作,将Bean B注入到Bean A中。

通过这样的三级缓存机制,Spring能够解决循环依赖的问题。当发生循环依赖时,早期实例的缓存earlySingletonObjects起到了临时存储的作用,保证了循环依赖的对象能够正确创建和注入。但需要注意的是,过多的循环依赖会增加系统的复杂性和性能开销,因此在设计应用程序时,应尽量避免循环依赖的出现。

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

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

相关文章

如何压缩高清PDF文件大小?将PDF文件压缩到最小的三个方法

PDF格式是一种非常常用的文档格式,但是有时候我们需要将PDF文件压缩为更小的大小以便于传输和存储。在本文中,我们将介绍三种PDF压缩的方法,包括在线PDF压缩、利用软件PDF压缩以及使用WPS缩小pdf。 首先,在线PDF压缩是最常用的方…

人体大脑神经元运行模拟器!让你直观体验大脑的运行方式

首先,宣布沾花把玖正式回归!!! 最近沾花在网上看到一个神奇的网站:A Neural Network Playground 经过沾花的亲手测试,发现这玩意儿能模拟人体大脑神经元的运行! 下面是网址: A N…

干货!机器视觉基础知识汇总

现如今,中国已经成为世界机器视觉发展最为活跃地区,应用范围涵盖了工业、农业、医药、军事、航天、气象等国民经济各个行业。虽然机器视觉的成长速度非常快,但是还是有很多人对机器视觉并不了解,今天我们来了解下机器视觉。 机器视觉就是用机器代替人眼来做测量和判断。机器…

一条自由游动的鲸鱼

先看效果&#xff1a; 再看代码&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>鲸鱼</title><style>#canvas-container {width: 100%;height: 100vh;overflow: hidden;}&l…

Linux(二)---------网络命令学习(ifconfig命令)

1.ifconfig命令 用于配置网卡ip地址信息&#xff0c;等网络参数信息&#xff0c;或者查看显示网络接口信息&#xff0c;类似于windows的ipconfig命令&#xff0c;还能够临时性的配置ip地址&#xff0c;子网掩码&#xff0c;广播地址&#xff0c;网关信息等。 注意ifconfig命令…

配置GIt账号、配置公钥

1.设置账号和邮箱 打开终端输入以下命令&#xff1a; git config --global --unset-all user.name git config --global --unset-all user.email然后输入以下命令来设置新的账号和邮箱&#xff1a; git config --global user.name "your_username" git config --glo…

整理了250个shell脚本,拿来即用!

无论是系统运维&#xff0c;还是应用运维&#xff0c;均可分为“纯手工”→ “脚本化”→ “自动化”→“智能化”几个阶段&#xff0c;其中自动化阶段&#xff0c;主要是将一些重复性人工操作和运维经验封装为程序或脚本&#xff0c;一方面避免重复性操作及风险&#xff0c;另…

【音视频SDK测评】线上K歌软件开发技术选型

摘要 在线K歌软件的开发有许多技术难点&#xff0c;需考虑到音频录制和处理、实时音频传输和同步、音频压缩和解压缩、设备兼容性问题等技术难点外&#xff0c;此外&#xff0c;开发者还应关注音乐版权问题&#xff0c;确保开发的应用合规合法。 前言 前面写了几期关于直播 …

qt系列-qt6在线安装慢的问题

.\qt-unified-windows-x64-online.exe --mirror https://mirrors.aliyun.com/qt/下载速度飞快

剑指offer19.正则表达式

这道题我一看就有印象&#xff0c;我室友算法课设抽到这题&#xff0c;他当时有个bug让我帮他看一下&#xff0c;然后我就大概看了一下他的算法&#xff0c;他是用动态规划写的&#xff0c;用了一个二维数组&#xff0c;然后我就试着按照这个思路去写&#xff0c;想了一会还是没…

输入筛选框搜索

文章目录 输入筛选框实现效果图需求前端工具版本添加依赖main.js导入依赖 代码 后端代码对应 sql对应 mapper.xml 文件的动态 sql 输入筛选框实现 效果图 需求 通过筛选框&#xff0c;选择公司&#xff0c;传入后端&#xff0c;后端根据公司名称去文章的内容中进行模糊查询 …

flask响应

介绍 在flask中&#xff0c;响应的方式有很多种&#xff0c;可以是普通字符串、json数据、html文本、模板或者是重定向。视图函数的返回值会自动转换为一个响应对象 当响应对象是字符串时 根据这个字符串和缺省参数自动生成一个用于返回的 响应对象 app.route("/test&q…

使用Debate Dynamics在知识图谱上进行推理(2020)7.31+8.1+8.2

使用Debate Dynamics在知识图谱上进行推理 摘要介绍背景与相关工作我们的方法状态action环境policiesDebate Dynamics裁判奖励报酬最大化和培训计划 实验数据集度量和评估方案结果 总结 摘要 我们提出了一种新的基于 Debate Dynamics 的知识图谱自动推理方法。 其主要思想是将…

2023年DevOps和云趋势报告!

要点 ●云创新已从革命性阶段转变为演进性阶段&#xff0c;重点是迁移和重新架构工作负载。云空间已发展为提供对可扩展资源和托管服务的按需访问&#xff0c;强调简化交互并减少团队的认知负担。 ●人工智能 (AI) 和大型语言模型 (LLM) 可以通过解决认知过载问题并支持即时管…

疯狂收割offer,全网最全接口测试面试题+答案,面试必刷题...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 面试题&#xff1…

网络安全公司校招面试会面试那些问题?

面试官会从那些方面去考察面试者&#xff1f; 以某安全公司的技术支持工程师岗位为例 面试官可能会从网络技术、操作系统、数据库、项目经验、语言表达以及个人擅长技能方面展开 面试官会提出那些问题来考查面试者呢&#xff1f; 网络基础方面的问题&#xff1a;请介绍一下…

Gitlab CI/CD笔记-第一天-GitOps和以前的和jenkins的集成的区别

一、GitOps-CI/CD的流程图 简单解释&#xff1a; 1.提交代码 2.编译构建 3.测试 4.部署 二、gitlab的实现 1、Runer 1.这个就是jenkins里的worker-slave的角色&#xff0c; 2.git-lab server 下发任务&#xff0c;Runner执行。 3.这个R…

JZ67 把字符串转换成整数(atoi)

目录 一、题目 二、易错点代码 一、题目 把字符串转换成整数(atoi)_牛客题霸_牛客网 (nowcoder.com) 二、易错点代码 int类型运算可能出现溢出的现象&#xff0c;因此可采用将int类型数据的运算转换成long long类型 class Solution { public:int is(long long result)//用…

Emacs之编译系统文件cc-mode.el.gz(一百二十四)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

leetcode每日一练-第108题-将有序数组转换为二叉搜索树

一、思路 递归 二、解题方法 在给定中序遍历序列数组的情况下&#xff0c;每一个子树中的数字在数组中一定是连续的&#xff0c;因此可以通过数组下标范围确定子树包含的数字&#xff0c;下标范围记为 [left,right]。对于整个中序遍历序列&#xff0c;下标范围从 left0到 ri…