Spring事务失效的全面剖析
文章目录
- 1. Spring事务基础
- 1.1 什么是Spring事务
- 1.2 Spring事务的实现原理
- 1.3 `@Transactional`注解的主要属性
- 1.4 使用Spring事务的简单示例
- 2. Spring事务失效的常见场景及解决方案
- 2.1 方法不是public的
- 问题描述
- 问题示例
- 解决方案
- 技术原理解释
- 2.2 自调用问题(同一个类中的方法调用)
- 问题描述
- 问题示例
- 解决方案
- 技术原理解释
- 2.3 异常被捕获而未被抛出
- 问题描述
- 问题示例
- 解决方案
- 技术原理解释
- 2.4 抛出的异常类型不正确
- 问题描述
- 问题示例
- 解决方案
- 技术原理解释
- 2.5 事务传播行为设置不当
- 问题描述
- 问题示例
- 解决方案
- 技术原理解释
- 2.6 未被Spring管理的类
- 问题描述
- 问题示例
- 解决方案
- 技术原理解释
- 2.7 数据库引擎不支持事务
- 问题描述
- 问题示例
- 解决方案
- 技术原理解释
- 2.8 使用了错误的事务管理器
- 问题描述
- 问题示例
- 解决方案
- 技术原理解释
- 2.9 多线程环境下的事务问题
- 问题描述
- 问题示例
- 解决方案
- 技术原理解释
- 2.10 代理的限制(CGLIB代理和final方法)
- 问题描述
- 问题示例
- 解决方案
- 技术原理解释
- 3. 常见问题与解答
- Q1: 为什么在同一个类中调用`@Transactional`方法会导致事务失效?
- Q2: Spring事务默认对哪些异常回滚?
- Q3: 为什么有时候抛出异常后事务没有回滚?
- Q4: `@Transactional`注解应该放在接口上还是实现类上?
- Q5: 有没有办法在一个事务中部分回滚?
- Q6: 如何在事务提交前或回滚后执行某些操作?
- Q7: 如何在测试中验证事务是否正常工作?
- 4. 最佳实践
- 4.1 设计和开发阶段
- 4.2 实现阶段
- 4.3 测试阶段
- 5. 总结
1. Spring事务基础
1.1 什么是Spring事务
事务是指对数据库执行的一系列操作,这些操作要么全部成功执行,要么全部不执行,以保证数据的一致性和完整性。Spring事务管理是Spring框架中一个强大的功能,它提供了一套完整的事务管理机制,使开发人员能够以声明式或编程式的方式管理事务。
Spring事务管理的核心是抽象出了一套事务管理的API,无论使用JDBC、Hibernate还是JPA,Spring都能以一致的方式提供事务支持。
1.2 Spring事务的实现原理
Spring事务的实现原理主要基于AOP(面向切面编程)和代理模式。当我们在一个方法上添加@Transactional
注解时,Spring会使用AOP创建一个代理对象,这个代理对象会在目标方法执行前开启事务,在方法执行后提交事务,如果发生异常则回滚事务。
具体流程如下:
- Spring容器初始化时,扫描带有
@Transactional
注解的方法 - 通过BeanPostProcessor创建代理对象(默认使用JDK动态代理,如果类没有实现接口则使用CGLIB代理)
- 当调用带有
@Transactional
注解的方法时,会首先执行代理对象的逻辑 - 代理对象负责开启事务、执行原方法、提交或回滚事务
1.3 @Transactional
注解的主要属性
@Transactional(propagation = Propagation.REQUIRED, // 事务传播行为isolation = Isolation.DEFAULT, // 事务隔离级别timeout = -1, // 事务超时时间readOnly = false, // 是否只读事务rollbackFor = Exception.class, // 遇到哪些异常回滚noRollbackFor = FileNotFoundException.class // 遇到哪些异常不回滚
)
1.4 使用Spring事务的简单示例
下面是一个基本的Spring事务使用示例:
@Service
public class UserService {@Autowiredprivate UserRepository userRepository;@Transactionalpublic void createUser(User user) {userRepository.save(user);// 如果后续操作抛出异常,该操作会回滚}
}
2. Spring事务失效的常见场景及解决方案
2.1 方法不是public的
问题描述
Spring官方文档明确指出,@Transactional
注解只能应用于public
方法上。如果将其应用于非public方法,该注解不会生效。
问题示例
@Service
public class UserService {@Autowiredprivate UserRepository userRepository;// 事务无效,因为方法不是public的@Transactionalprotected void createUser(User user) {userRepository.save(user);throw new RuntimeException("故意抛出异常");}
}
解决方案
确保所有使用@Transactional
注解的方法都是public
的。
@Service
public class UserService {@Autowiredprivate UserRepository userRepository;// 正确使用,方法是public的@Transactionalpublic void createUser(User user) {userRepository.save(user);// 如果抛出异常,事务会回滚}
}
技术原理解释
Spring的事务管理是通过AOP实现的,而Spring AOP的默认行为是只拦截public
方法。这是因为在代理对象中,非public
方法无法被外部调用,所以Spring AOP也就无法为其创建代理。
2.2 自调用问题(同一个类中的方法调用)
问题描述
当一个类中的方法调用同一个类中的另一个事务方法时,事务不会生效。这是因为在这种情况下,调用的是目标对象的方法,而不是代理对象的方法。
问题示例
@Service
public class UserService {@Autowiredprivate UserRepository userRepository;// 事务生效的方法@Transactionalpublic void createUserWithRoles(User user, List<Role> roles) {// 直接调用同类中的另一个事务方法this.createUser(user); // 这里的调用不会使用事务代理// 添加角色逻辑for (Role role : roles) {userRepository.saveUserRole(user.getId(), role.getId());}// 如果这里抛出异常,createUser方法不会回滚throw new RuntimeException("故意抛出异常");}@Transactionalpublic void createUser(User user) {userRepository.save(user);}
}
在上面的例子中,如果createUserWithRoles
方法抛出异常,通过调用this.createUser(user)
创建的用户不会回滚,因为这是一个自调用,没有通过Spring的事务代理。
解决方案
有以下几种解决方案:
- 将方法移到其他类中:
@Service
public class UserService {@Autowiredprivate UserRepository userRepository;@Autowiredprivate RoleService roleService;@Transactionalpublic void createUserWithRoles(User user, List<Role> roles) {// 调用其他类的方法,事务会正常工作roleService.createUser(user);for (Role role : roles) {userRepository.saveUserRole(user.getId(), role.getId());}throw new RuntimeException("故意抛出异常");}
}@Service
public class RoleService {@Autowiredprivate UserRepository userRepository;@Transactionalpublic void createUser(User user) {userRepository.save(user);}
}
- 使用自我注入:
@Service
public class UserService {@Autowiredprivate UserRepository userRepository;// 自我注入,获取代理对象@Autowiredprivate UserService self;@Transactionalpublic void createUserWithRoles(User user, List<Role> roles) {// 通过代理对象调用方法,事务会正常工作self.createUser(user);for (Role role : roles) {userRepository.saveUserRole(user.getId(), role.getId());}throw new RuntimeException("故意抛出异常");}@Transactionalpublic void createUser(User user) {userRepository.save(user);}
}
- 使用AopContext获取代理对象(需要配置
exposeProxy = true
):
@EnableAspectJAutoProxy(exposeProxy = true)
@SpringBootApplication
public class Application {public static void main(String[] args) {SpringApplication.run(Application.class, args);}
}@Service
public class UserService {@Autowiredprivate UserRepository userRepository;@Transactionalpublic void createUserWithRoles(User user, List<Role> roles) {// 获取当前代理对象,调用其方法((UserService) AopContext.currentProxy()).createUser(user);for (Role role : roles) {userRepository.saveUserRole(user.getId(), role.getId());}throw new RuntimeException("故意抛出异常");}
相关文章:
Spring事务失效的全面剖析
文章目录 1. Spring事务基础1.1 什么是Spring事务1.2 Spring事务的实现原理1.3 `@Transactional`注解的主要属性1.4 使用Spring事务的简单示例2. Spring事务失效的常见场景及解决方案2.1 方法不是public的问题描述问题示例解决方案技术原理解释2.2 自调用问题(同一个类中的方法…...
Pytorch张量和损失函数
文章目录 张量张量类型张量例子使用概率分布创建张量正态分布创建张量 (torch.normal)正态分布创建张量示例标准正态分布创建张量标准正态分布创建张量示例均匀分布创建张量均匀分布创建张量示例 激活函数常见激活函数 损失函数(Pytorch API)L1范数损失函数均方误差损失函数交叉…...
消息~组件(群聊类型)ConcurrentHashMap发送
为什么选择ConcurrentHashMap? 在开发聊天应用时,我们需要存储和管理大量的聊天消息数据,这些数据会被多个线程频繁访问和修改。比如,当多个用户同时发送消息时,服务端需要同时处理这些消息的存储和查询。如果用普通的…...

FFmpeg多路节目流复用为一路包含多个节目的输出流
在音视频处理领域,将多个独立的节目流(如不同频道的音视频内容)合并为一个包含多个节目的输出流是常见需求。FFmpeg 作为功能强大的多媒体处理工具,提供了灵活的流复用能力,本文将通过具体案例解析如何使用 FFmpeg 实现…...

分子动力学模拟揭示点突变对 hCFTR NBD1结构域热稳定性的影响
囊性纤维化(CF) 作为一种严重的常染色体隐性遗传疾病,全球约有 10 万名患者深受其害。它会累及人体多个器官,如肺部、胰腺等,严重影响患者的生活质量和寿命。CF 的 “罪魁祸首” 是 CFTR 氯离子通道的突变,…...

关于SIS/DCS点检周期
在中国化工行业,近几年在设备维护上有个挺有意思的现象,即SIS和DCS这两个系统的点检周期问题,隔三差五就被管理层会议讨论,可以说是企业管理层关注的重要方向与关心要素。 与一般工业行业中设备运维不同,SIS与DCS的点…...
python-pyqt6框架工具开发总结
菜单栏 工具栏 状态栏 QTreeView 用于展示树形结构数据(模-视图框架),文件系统,组织结构 通常与QAbstractItemModel的子类(如QStandardItemModel或自动义模型) 展开/折叠 控制节点的显示状态,展开时显示子节点,折叠时隐藏子节点 s…...
Docker Volumes
Docker Volumes 是 Docker 提供的一种机制,用于持久化存储容器数据。与容器的生命周期不同,Volumes 可以独立存在,即使容器被删除,数据仍然保留。以下是关于 Docker Volumes 的详细说明: 1. 为什么需要 Volumes&#…...

【PmHub后端篇】PmHub中基于Redis加Lua脚本的计数器算法限流实现
1 限流的重要性 在高并发系统中,保护系统稳定运行的关键技术有缓存、降级和限流。 缓存通过在内存中存储常用数据,减少对数据库的访问,提升系统响应速度,如浏览器缓存、CDN缓存等多种应用层面。降级则是在系统压力过大或部分服务…...
FPGA实战项目2———多协议通信控制器
1. 多协议通信控制器模块 (multi_protocol_controller) 简要介绍 这是整个设计的顶层模块,承担着整合各个子模块的重要任务,是整个系统的核心枢纽。它负责协调 UART、SPI、I2C 等不同通信协议模块以及 DMA 模块的工作,同时处理不同时钟域之间的信号交互,确保各个模块能够…...

CST软件仿真案例——太阳能薄膜频谱吸收率
CST软件中的太阳能薄膜的功率吸收可用光频电磁波在介质材料中的损耗来计算。本案例计算非晶硅的功率吸收,然后考虑真实太阳频谱,计算有效吸收频谱。 用太阳能单元模板,时域求解器: 材料库提取四个材料,非晶硅…...
多线程进阶核心知识详解(通俗版)
Java多线程进阶详解 一、锁策略:如何高效管理资源竞争 在多线程环境中,锁是协调资源访问的核心机制。不同的锁策略适用于不同的场景,理解它们的差异能帮助优化程序性能。 1. 乐观锁 vs 悲观锁 悲观锁: 核心思想:假设…...
大模型中的KV Cache
1. KV Cache的定义与核心原理 KV Cache(Key-Value Cache)是一种在Transformer架构的大模型推理阶段使用的优化技术,通过缓存自注意力机制中的键(Key)和值(Value)矩阵,避免重复计算&…...
FHQ平衡树
FHQ平衡树 大致是这样的题目: 您需要动态地维护一个可重集合 M M M,并且提供以下操作: 向 M M M 中插入一个数 x x x。从 M M M 中删除一个数 x x x(若有多个相同的数,应只删除一个)。查询 M M M 中…...
力扣算法---总结篇
5.13 数组总结 数组是存放在连续内存空间上的相同类型数据的集合。 数组可以方便的通过下标索引的方式获取到下标对应的数据。 正是因为数组在内存空间的地址是连续的,所以我们在删除或者增添元素的时候,就难免要移动其他元素的地址。 数组的元素是不…...

ABAP+旧数据接管的会计年度未确定
导资产主数据时,报错旧数据接管的会计年度未确定 是因为程序里面使用了下列函数AISCO_CALCULATE_FIRST_DAY,输入公司代码,获取会计年度,这个数据是在后台表T093C表中取数的,通过SE16N可以看到后台表数据没有数…...
Java【10_1】用户注册登录(面向过程与面向对象)
测试题 1、基于文本界面实现登录注册的需求(要求可以满足多个用户的注册和登录) 通过工具去完成 公共类: public class User { private int id;//用户编号 private int username;//用户名 private int password;//密码 private String name;//真…...

养生:打造健康生活的全方位策略
在生活节奏不断加快的当下,养生已成为提升生活质量、维护身心平衡的重要方式。从饮食、运动到睡眠,再到心态调节,各个方面的养生之道共同构建起健康生活的坚实基础。以下为您详细介绍养生的关键要点,助您拥抱健康生活。 饮食养生…...

贪吃蛇游戏排行榜模块开发总结:从数据到视觉的实现
一、项目背景与成果概览 在完成贪吃蛇游戏核心玩法后,本次开发重点聚焦于排行榜系统的实现。该系统具备以下核心特性: 🌐 双数据源支持:本地存储(localStorage)与远程API自由切换 🕒 时间维度统计:日榜/周榜/月榜/全时段数据筛选 🎮 模式区分:闯关模式(关卡进度…...
pytorch 数据预处理和常用工具
文章目录 NumPyNumpy数据结构安装和使用NumPy Matplotlib的安装和导入安装和导入Matplotlib绘制基础图画折线图散点图柱状图图例 数据清洗据清洗的作用Pandas进行数据清洗Pandas数据结构Series 数据结构DataFrame数据结构 Pandas数据清洗常用代码 特征工程主成分分析线性判别分…...
如何界定合法收集数据?
首席数据官高鹏律师团队 在当今数字化时代,数据的价值日益凸显,而合法收集数据成为了企业、机构以及各类组织必须严守的关键准则。作为律师,深入理解并准确界定合法收集数据的范畴,对于保障各方权益、维护法律秩序至关重要。 一…...
企业对数据集成工具的需求及 ETL 工具工作原理详解
当下,数据已然成为企业运营发展过程中的关键生产要素,其重要性不言而喻。 海量的数据分散在企业的各类系统、平台以及不同的业务部门之中,企业要充分挖掘这些数据背后所蕴含的巨大价值,实现数据驱动的精准决策,数据集…...
内核深入学习3——分析ARM32和ARM64体系架构下的Linux内存区域示意图与页表的建立流程
内核深入学习3——ARM32/ARM64在Linux内核中的实现(2) 今天我们来讨论的是一个硬核的内容,也是一个老生常谈的话题——那就是分析ARM32和ARM64体系架构下的Linux内存区域示意图的内容。对于ARM64的部分,我们早就知道一个基本的…...
MapReduce基本介绍
核心思想 分而治之:将大规模的数据处理任务分解成多个可以并行处理的子任务,然后将这些子任务分配到不同的计算节点上进行处理,最后将各个子任务的处理结果合并起来,得到最终的结果。 工作流程 Map 阶段: 输入数据被…...

屏幕与触摸调试
本章配套视频介绍: 《28-屏幕与触摸设置》 【鲁班猫】28-屏幕与触摸设置_哔哩哔哩_bilibili LubanCat-RK3588系列板卡都支持mipi屏以及hdmi显示屏的显示。 19.1. 旋转触摸屏 参考文章 触摸校准 参考文章 旋转触摸方向 配置触摸旋转方向 1 2 # 1.查看触摸输入设备 xinput…...

使用 百度云大模型平台 做 【提示词优化】
1. 百度云大模型平台 百度智能云千帆大模型平台  平台功能:演示了阿里云大模型的百炼平台,该平台提供Prompt工程功能,支持在线创建和优化Prompt模板模板类型:平台提供多种预制模板,同时也支持用户自定义…...
C 语言_常见排序算法全解析
排序算法是计算机科学中的基础内容,本文将介绍 C 语言中几种常见的排序算法,包括实现代码、时间复杂度分析、适用场景和详细解析。 一、冒泡排序(Bubble Sort) 基本思想:重复遍历数组,比较相邻元素,将较大元素交换到右侧。 代码实现: void bubbleSort(int arr[], i…...

IJCAI 2025 | 高德首个原生3D生成基座大模型「G3PT」重塑3D生成的未来
国际人工智能联合会议(IJCAI)是人工智能领域最古老、最具权威性的学术会议之一,自1969年首次举办以来,至今已有近六十年的历史。它见证了人工智能从萌芽到蓬勃发展的全过程,是全球人工智能研究者、学者、工程师和行业专…...

Samtec助力电视广播行业
【摘要前言】 现代广播电视技术最有趣的方面之一就是界限的模糊。过去,音频和视频是通过射频电缆传输的模拟技术采集的,而现在,数字世界已经取代了模拟技术。物理胶片和磁带已让位于数字存储设备和流媒体。 在这个过程中,连接器…...

密码学--仿射密码
一、实验目的 1、通过实现简单的古典密码算法,理解密码学的相关概念 2、理解明文、密文、加密密钥、解密密钥、加密算法、解密算法、流密码与分组密码等。 二、实验内容 1、题目内容描述 ①随机生成加密密钥,并验证密钥的可行性 ②从plain文件读入待…...