@Transactional导致的循环依赖问题BeanNotOfRequiredTypeException
创始人
2025-05-31 03:02:44

首先我有一个Class A和Class B,A和B存在循环依赖。

@Service
@Transactional(rollbackFor = Exception.class)
public class A implements Ainterface{@Autowiredprivate B b;@Overridepublic void methodA() {// do something...}
}@Service
public class B implements BInterface{@Autowiredprivate A a;@Overridepublic void methodB() {a.methodA();}
}

一般情况下,也就是不加Transactional注解是没有问题的,因为我们的spring框架默认就支持循环依赖,但是我们这个例子在Class A上加了个注解,为什么会有循环依赖的报错呢,报错信息如下:
org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name ‘b’: Unsatisfied dependency expressed through field ‘a’; nested exception is org.springframework.beans.factory.BeanNotOfRequiredTypeException: Bean named ‘a’ is expected to be of type ‘com.qhyu.cloud.circlarRefrence.A’ but was actually of type ‘com.sun.proxy.$Proxy32’

该异常信息表明在创建名为 ‘b’ 的Bean时,其依赖的 ‘a’ 字段无法满足依赖关系,因为Spring无法将 ‘a’ 注入到 ‘b’ 中的 ‘a’ 字段上。

异常信息中还提到了一个嵌套异常,即 ‘BeanNotOfRequiredTypeException’,它表明 Spring 无法将名为 ‘a’ 的Bean注入到 ‘b’ 中的 ‘a’ 字段上,因为 ‘a’ Bean的类型与 ‘b’ 中 ‘a’ 字段的类型不匹配。具体来说, ‘a’ Bean的实际类型是 ‘com.sun.proxy.$Proxy32’,而 ‘b’ 中的 ‘a’ 字段的期望类型是 ‘com.qhyu.cloud.circlarRefrence.A’。

通过这个异常信息,我们很快就找到了思路,初步估计问题是发生在populateBean方法,在属性注入的时候会验证属性类型,因为我们的Service都是实现了接口的,所以在没有强制使用cglb进行代理的情况下,会使用jdk动态代理,当存在代理对象时,如果使用JDK动态代理,代理对象会实现目标接口并继承Proxy类,而不是原始类,因此在填充代理对象的属性时,Spring无法确定该使用哪一个对象进行属性填充,因此无法正确地替换代理对象。这种情况下,可以考虑使用CGLIB动态代理来解决该问题。

源码中校验的位置在DefaultListableBeanFactory的1401行,报错信息在1402行抛出。
在这里插入图片描述

要解决这个问题,可以采用以下方法:

1、解决循环依赖:尽量避免类之间的循环依赖,可以通过将共享的方法或数据移动到一个新的类中,然后让两个类分别依赖这个新类。

2、使用setter注入:如果无法避免循环依赖,可以考虑使用setter方法进行依赖注入,而不是构造方法注入。这样,当一个类被实例化时,它的依赖关系会在实例化之后才被注入,可以避免代理对象替换的问题。

3、指定代理类型:可以明确指定代理类型。例如,可以使用@EnableTransactionManagement(proxyTargetClass = true)来强制使用基于类的代理(CGLIB代理),而不是默认的基于接口的代理(JDK动态代理)。这样,循环依赖解析过程中,代理对象会被正确地替换。或者将@Transactional 注解移动到方法上。

相关内容

热门资讯

4.线性表的循环链表、双向链表... 数据结构很重要! 数据结构很重要!!! 数据...
Spark - 继承 File... 目录 一.引言 二.源码浅析 1.RDD.saveAsTextFile 2.TextOutputFo...
人工智能和网络安全,应该如何选... 随着数字时代的到来,网络安全和人工智能成了科技创新产业的重要组成部分。也逐渐成了大多数...
Java面试总结(九) redis的持久化方式 RDB、AOF、混合。 RDB 持久化 Redis 可以通过创建快照来获得存...
pytest学习和使用22-a... 22-allure特性 丨总览中的Environment和Categories设置1 Environ...
黄鹤使用说明书-主动给下级当好... 几年前《xx说明书》这类书籍大行其道的时候,就想过写一个自己的使用说明书,...
Ripser.py学习 (8)... 文章目录1 概述2 测试数据 1 概述 本节展示系数域如何影响H1H_1H1​同伦。这个例子...
【真八股 | 华为OD技术面试... 文章目录 华为 OD 面试流程1. Java 都有哪些锁2. 各种设计模式3. 如何打开一个文件并从...
day05_Java中的运算符 在Java中提供了丰富的运算符  其按照功能分:算术运算符、赋值运算符、比较运算符、逻...
为什么热咖啡保温几小时后的变化... 为什么热咖啡保温几小时后的变化比冰咖啡大?导读心理影响感官的作用咖啡的化学变化 导读 ...
总结796 早上: 7:10起床 7:35~8:00背单词(第一遍) ...
Pytorch深度学习常用预训... Resnet:model_urls = {‘resnet18’: ‘https://down...
Python之面相对象语法全解 1、类的定义和使用 - 类和对象都是对现实生活中的事物或程序中的内容的抽象 - 实际上所有的事物都由...
Maven高级——聚合、继承、... 一、分模块开发与设计直接引用前面写过的https://blog.csdn.net/weixin_51...
简单数据查询 一、实验目的       熟悉和掌握对数据表中数据的查询操作和SQL命令的使用,学会灵...
WuThreat身份安全云-T... 漏洞名称: Ubiquiti EdgeRouter 命令注入漏洞 漏洞级别:中危 漏洞编号:CVE-...
C++类和对象(上篇) 目录 1.类的定义 2.类的访问限定符及封装   2.1类的访问限定符   2.2封装 3.类的作用...
社区最早一批贡献者为什么还在坚... 点击蓝字 关注我们今天这位我们将要介绍的项目 PMC Member 是社区最早的一批贡献者之一&#x...
文心一言 vs GPT-4 —... 文心一言 vs GPT-4 —— 全面横向比较 3月15日凌晨,OpenAI发布“迄今...
element-ui表单先编辑... element-ui的表单resetFields()方法无法清空 需要注意几个问题: ...
Jemeter测试--安装与使... Jmeter简介 JMeter,一个100%的纯Java桌面应用...
OCR之论文笔记TrOCR 文章目录TrOCR: Transformer-based Optical Character Rec...
Mysql常用数据类型总结 整形 枚举类型ENUE整形       TINYINT,SMALLINT,MEDIUMINT,IN...
【flink sql】创建表 flink sql创建表语法 CREATE TABLE [IF NOT EXISTS] [catal...
python opencv 保... 👨‍💻个人简介: 深度学习图像领域工作者 dz...
Pytorch深度学习实战3-... 目录1 数据集Dataset2 数据加载DataLoader3 常用预处理方法4 模型处理5 实例&...
自定义类型的超详细讲解ᵎᵎ了解...   目录 1.结构体的声明 1.1基础知识 1.2结构体的声明 1.3结构体的特殊声明  1.4结构...
Docker等容器技术如何与移... 移动应用程序的开发面临着很多挑战,包括开发环境的设置、测试的困难、部署的复杂性等。由于...
【微服务】—— Nacos安装... 文章目录1. Windows安装1.1 下载安装包1.2 解压1.3 端口配置1.4 启动1.5 访...
【OpenGL】 为了理解这个函数我们需要先学习一些OpenGL的内容 OpenGL可视化 https://g...