深度剖析Spring循环依赖(实战Bug)

目录

  • 前言
  • 1. 问题所示
  • 2. 原理分析
  • 3. 基本知识
  • 4. @Lazy注解

前言

通过实战更好的回馈问题,意识更加深刻

起因是我出现如下问题之后,才意识到中了Spring的循环依赖了!

1. 问题所示

在执行项目的时候,出现如下问题,问题如下所示:

org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'tyreRepareOrderController' defined in file [F:\java_project\xx\xx-service\xx-equipment\target\classes\org\xx\equipment\controller\TyreRepareOrderController.class]: Unsatisfied dependency expressed through constructor parameter 0; nested exception is org.springframework.beans.factory.BeanCurrentlyInCreationException: Error creating bean with name 'tyreRepareOrderServiceImpl': Bean with name 'tyreRepareOrderServiceImpl' has been injected into other beans [tyreRepareOrderServiceImpl] in its raw version as part of a circular reference, but has eventually been wrapped. This means that said other beans do not use the final version of the bean. This is often the result of over-eager type matching - consider using 'getBeanNamesForType' with the 'allowEagerInit' flag turned off, for example.at org.springframework.beans.factory.support.ConstructorResolver.createArgumentArray(ConstructorResolver.java:769)at org.springframework.beans.factory.support.ConstructorResolver.autowireConstructor(ConstructorResolver.java:218)at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.autowireConstructor(AbstractAutowireCapableBeanFactory.java:1338)at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBeanInstance(AbstractAutowireCapableBeanFactory.java:1185)

截图如下所示:

在这里插入图片描述

2. 原理分析

这是Spring循环依赖的经典问题了

原本我是ServiceImpl类中实现了Service中的方法!
之后又在ServiceImpl类 通过 注入 Service方法来操作,说的普通一点就是死锁了!

解决方案就是要么使用Spring中的三级缓存(这是面试的经典题)
或者使用this.super直接使用,从而不用再次注入,而又能调用其方法!

3. 基本知识

Spring循环依赖是指两个或多个Bean彼此依赖,形成一个循环引用的情况。这种情况可能导致应用程序无法正常启动或运行,因此需要特殊处理。

在Spring中,循环依赖的处理是通过使用三级缓存解决的。三级缓存是指Spring容器中的三个缓存区域,分别是singletonObjects、earlySingletonObjects和singletonFactories。在处理循环依赖时,Spring会通过提前暴露半成品Bean的方式来解决。

具体来说,当一个Bean被创建时,Spring会将其提前暴露给singletonFactories缓存,即使它还没有完全初始化。这样,其他Bean在需要引用这个Bean时,可以通过singletonFactories获取到它的引用,而不是等待其完全初始化。当Bean初始化完成后,会将其放入singletonObjects缓存,表示完全初始化的Bean。

这样,循环依赖中的Bean就能够在初始化过程中相互引用,而不会导致死循环或空指针异常。

需要注意的是,Spring的循环依赖处理机制对于单例(singleton)的Bean是有效的,对于其他作用域的Bean(如prototype)可能会有不同的处理方式。

下面是一个简单的Java代码示例,演示了Spring中的循环依赖情况:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Component
public class A {@Autowiredprivate B b;// Other code...
}@Component
public class B {@Autowiredprivate A a;// Other code...
}

在上述例子中,类A依赖类B,而类B又依赖类A,形成了循环依赖。

Spring会通过三级缓存解决这种循环依赖。当一个Bean被创建时,它会被提前暴露给singletonFactories缓存,以便其他Bean可以引用它。下面是如何解除循环依赖的代码解释:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Component;@Component
public class A {@Autowired@Lazyprivate B b;// Other code...
}@Component
public class B {@Autowired@Lazyprivate A a;// Other code...
}

在解除循环依赖时,可以使用@Lazy注解标记依赖关系,表示在需要时才进行实际的Bean初始化。

这样,当Bean A和Bean B相互引用时,它们的初始化将被延迟到第一次实际使用时。这有助于打破循环依赖,因为Bean的初始化被推迟到真正需要的时候。

4. @Lazy注解

@Lazy注解是Spring框架提供的一种机制,用于延迟加载(Lazy Initialization)Bean。

当一个Bean被标记为@Lazy时,它的初始化将会被推迟到第一次被实际使用的时候,而不是在容器启动时立即初始化。

延迟加载的一些补充:

优点作用范围使用场景
1.性能优化: 避免在应用启动时就初始化所有Bean,节省了启动时间和资源。

2.解决循环依赖: 通过延迟加载,可以解决某些情况下的循环依赖问题,如前面提到的循环依赖的处理。
@Lazy注解可以用于标记在字段、方法、构造函数或配置类(@Configuration)的@Bean方法上。适用于那些在启动时不一定会被立即使用的Bean,特别是当Bean的初始化过程较为耗时或涉及到一些外部资源。

与单例(Singleton)的关系: 默认情况下,Spring容器中的Bean是单例的。
使用@Lazy注解可以将单例Bean的初始化延迟到第一次使用时。

XML配置中的等效方式: 在XML配置中,可以使用lazy-init="true"属性来实现与@Lazy注解相同的效果。

<bean id="myBean" class="com.example.MyBean" lazy-init="true"><!-- Bean configuration -->
</bean>

总体而言,@Lazy注解是Spring框架提供的一个有用的特性,可用于优化应用程序的启动性能,尤其是在处理大型应用或有复杂依赖关系的情况下。

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

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

相关文章

面试之Glide如何绑定Activity的生命周期

Glide绑定Activity生命周期 Glide.with() 下面都是它的重载方法&#xff0c;Context&#xff0c;Activity&#xff0c;FragmentActivity, Fragment, android.app.Fragment fragment,View都可以作为他的参数&#xff0c;内容大同小异&#xff0c;都是先getRetriever&#xff0…

016-Vue-黑马2023:前后端分离开发(在线接口文档),前端工程化、Element、vue编写一个完成页面、Vue路由、vue打包部署到nginx

第三节 前后端分离开发 1、介绍 开发模式 前后端混合开发&#xff1a;传统开发模式 前后端分离开发&#xff1a;当前最为主流的开发模式 页面原型需求案例&#xff1a;分析出接口文档 离线开发文档示例&#xff1a; 2、YAPI&#xff08;官网已停用&#xff09; 202…

汽车微电机行业研究:预计2029年将达到188亿美元

微电机行业是技术密集型行业&#xff0c;其起源于欧洲的德国、瑞士等国家&#xff0c;发展于日本。随着改革开放&#xff0c;中国作为发展中国家&#xff0c;承接了德国、日本等发达国家的汽车微电机产业转移&#xff0c;技术扩散逐步向我国转移。 微特电机广泛应用于信息处理设…

优化微信小程序更新体验:异步更新与强制更新方案解析

在微信小程序的开发和迭代过程中&#xff0c;新版本覆盖率的问题一直备受关注。由于小程序采用异步更新机制&#xff0c;在用户首次打开或冷启动时才会检查并下载新版本&#xff0c;导致部分用户无法及时应用上最新版本。为了解决这一问题&#xff0c;微信团队经过深入研究和讨…

设计社交网络的数据结构

1: 确定 Use Case 和 约束 Use Cases User 搜索某人然后看到被搜索人的最短路径Service 有高可用 约束和假设 状态假设 Traffic 不是平均分布的 一些被搜索者是更加受欢迎的&#xff0c;某些被搜索者只会被搜索一次图数据不适用与单个机器图的分布是轻量级的一亿个 User每…

canvas绘制N角形,锯齿状

查看专栏目录 canvas实例应用100专栏&#xff0c;提供canvas的基础知识&#xff0c;高级动画&#xff0c;相关应用扩展等信息。canvas作为html的一部分&#xff0c;是图像图标地图可视化的一个重要的基础&#xff0c;学好了canvas&#xff0c;在其他的一些应用上将会起到非常重…

在线扒站网PHP源码-在线扒站工具网站源码

源码介绍 这是一款在线的网站模板下载程序&#xff0c;也就是我们常说的扒站工具&#xff0c;利用它我们可以很轻松的将别人的网站模板样式下载下来&#xff0c;这样就可以大大提高我们编写前端的速度了&#xff01;注&#xff1a;扒取的任何站点不得用于商业、违法用途&#…

kafka参数配置参考和优化建议 —— 筑梦之路

对于Kafka的优化&#xff0c;可以从以下几个方面进行思考和优化&#xff1a; 硬件优化&#xff1a;使用高性能的硬件设备&#xff0c;包括高速磁盘、大内存和高性能网络设备&#xff0c;以提高Kafka集群的整体性能。 配置优化&#xff1a;调整Kafka的配置参数&#xff0c;包括…

免费的爬虫软件【2024最新】

在国际市场竞争日益激烈的背景下&#xff0c;国外网站的SEO排名直接关系到网站在搜索引擎中的曝光度和用户点击量。良好的SEO排名能够带来更多的有针对性的流量&#xff0c;提升网站的知名度和竞争力。 二、国外网站SEO排名的三种方法 关键词优化&#xff1a; 关键词优化是SEO…

Red Hat Enterprise Linux 9.3 安装图解

引导和开始安装 选择倒计时结束前&#xff0c;通过键盘上下键选择下图框选项&#xff0c;启动图形化安装过程。需要注意的不同主板默认或者自行配置的固件类型不一致&#xff0c;引导界面有所不同。也就是说使用UEFI和BIOS的安装引导界面是不同的&#xff0c;如图所示。若手动调…

ITK + ANT,无法显示三维

背景&#xff1a;之前用ANT保存ima格式的数据&#xff0c;选择的是保存所有的序列 用python将dicom转为nii的格式&#xff0c; import nibabel as nib import torch"""不管是nii还是nii.gz都是二维的&#xff0c;为啥呢"""fobj nib.load("…

Qt弹框展示

1.相关说明 文件选择弹框、目录选择弹框、保存文件弹框、颜色选择弹框、字体选择弹框、进度条弹框、输入对话框、标准消息框等 2.相关界面 3.相关代码 #include "widget.h" #include "ui_widget.h" #include <QFileDialog> #include <QProgressD…

Red Hat Enterprise Linux 6.10 安装图解

引导和开始安装 选择倒计时结束前&#xff0c;通过键盘上下键选择下图框选项&#xff0c;启动图形化安装过程。需要注意的不同主板默认或者自行配置的固件类型不一致&#xff0c;引导界面有所不同。也就是说使用UEFI和BIOS的安装引导界面是不同的&#xff0c;如图所示。若手动调…

go 语言(九)----struct

定义一个结构体 type Book struct {title stringauth string }结构体使用 package mainimport "fmt"//定义一个结构体 type Book struct {title stringauth string }func main() {var book1 Bookbook1.title "Golang"book1.auth "zhang3"fmt…

使用 TiUP 部署 TiDB 集群

TIDB优点 支持分布式且支持事务的关系型数据库&#xff0c;不用考虑分库分表 同时满足了可伸缩&#xff0c;高可用&#xff0c;关系型&#xff0c;支持事务。 基本上按官网的文档来就行了。 在线部署 以普通用户身份登录中控机。以 tidb 用户为例&#xff0c;后续安装 TiUP …

用git bash调用md5sum进行批量MD5计算

对于非常大的文件或者很重要的文件&#xff0c;在不稳定的网络环境下&#xff0c;可能文件的某些字节会损坏。此时&#xff0c;对文件计算MD5即可以校验其完整性。比如本次的 OpenStreetMap 导出包&#xff0c;我的学弟反馈通过网盘下载无法解压&#xff0c;并建议我增加每个文…

MOS管和IGBT管的定义与辨别

MOS管和IGBT管作为现代电子设备使用频率较高的新型电子器件&#xff0c;因此在电子电路中常常碰到也习以为常。可是MOS管和IGBT管由于外形及静态参数相似的很&#xff0c;有时在选择、判断、使用容易出差池。MOS管和IGBT管可靠的识别方法为选择、判断、使用扫清障碍&#xff01…

android使用相机 intent.resolveActivity returns null

问题 笔者使用java进行android开发&#xff0c;启动相机时 intent.resolveActivity returns null takePictureIntent.resolveActivity(getPackageManager()) null详细问题 笔者使用如下代码启动相机 // 启动相机SuppressLint("LongLogTag")private void dispatc…

【LGR-172-Div.4】洛谷入门赛 #19(A—H,c++详解!)

文章目录 【LGR-172-Div.4】洛谷入门赛 #19A.分饼干 I题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 样例 #2样例输入 #2样例输出 #2 提示样例解释 1样例解释 2数据范围与约定思路: 代码 B.分饼干 II题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 样例 #2样…

2024--Django平台开发-订单项目管理(十四)

day14 订单管理系统 1.关于登录 1.1 UI美化 页面美化&#xff0c;用BootStrap 自定义BooStrapForm类实现。 class BootStrapForm:exclude_filed_list []def __init__(self, *args, **kwargs):super().__init__(*args, **kwargs)# {title:对象,"percent":对象}fo…