SpringData JPA事务管理:@Transactional注解与事务传播

文章目录
- 引言
- 一、事务基础概念
- 二、@Transactional注解详解
- 2.1 基本用法
- 2.2 属性配置
- 2.3 类级别与方法级别
- 三、事务传播行为详解
- 3.1 REQUIRED(默认)
- 3.2 REQUIRES_NEW
- 3.3 其他传播行为
- 四、事务隔离级别
- 五、事务最佳实践
- 5.1 正确设置事务边界
- 5.2 合理使用只读事务
- 5.3 选择合适的隔离级别
- 总结
引言
在企业级Java应用开发中,事务管理是确保数据一致性和完整性的关键机制。Spring Data JPA提供了强大而灵活的事务管理功能,通过声明式注解简化了复杂事务的实现。本文将深入解析@Transactional注解的使用方法以及事务传播行为的各种模式,帮助开发者掌握如何在实际应用中正确使用事务,确保数据操作符合ACID特性(原子性、一致性、隔离性和持久性)。
一、事务基础概念
事务是数据库操作的基本单位,它确保一组相关的数据库操作要么全部成功,要么全部失败。Spring框架通过抽象层统一了不同数据访问技术的事务管理,为Spring Data JPA提供了一致的事务支持。
Spring事务管理的核心是事务管理器(PlatformTransactionManager),它负责事务的开始、提交和回滚。Spring Data JPA默认使用JpaTransactionManager作为事务管理器,与底层的JPA实现(如Hibernate)进行集成。
@Configuration
@EnableTransactionManagement
public class TransactionConfig {@Beanpublic PlatformTransactionManager transactionManager(EntityManagerFactory entityManagerFactory) {JpaTransactionManager transactionManager = new JpaTransactionManager();transactionManager.setEntityManagerFactory(entityManagerFactory);return transactionManager;}
}
二、@Transactional注解详解
@Transactional是Spring框架提供的核心注解,用于声明事务边界。在Spring Data JPA中,它可以应用于类和方法级别,提供了丰富的配置选项。
2.1 基本用法
@Transactional注解的基本用法是将其应用于需要事务支持的方法或类:
@Service
public class OrderService {@Autowiredprivate OrderRepository orderRepository;@Autowiredprivate OrderItemRepository orderItemRepository;/*** 创建订单并保存订单项* 整个方法作为一个事务执行*/@Transactionalpublic Order createOrder(Order order, List<OrderItem> items) {// 保存订单Order savedOrder = orderRepository.save(order);// 保存订单项for (OrderItem item : items) {item.setOrder(savedOrder);orderItemRepository.save(item);}return savedOrder;}
}
2.2 属性配置
@Transactional注解提供了多个属性,用于配置事务的行为:
@Transactional(timeout = 30, // 事务超时时间(秒)readOnly = false, // 非只读事务isolation = Isolation.READ_COMMITTED, // 隔离级别propagation = Propagation.REQUIRED, // 传播行为rollbackFor = Exception.class // 发生异常时回滚
)
public Payment processPayment(Payment payment, Long accountId, double amount) {// 事务操作
}
主要属性说明:
- timeout:指定事务的超时时间,超过该时间事务将自动回滚
- readOnly:标记事务是否为只读,只读事务可以优化性能
- isolation:指定事务的隔离级别,控制并发事务的可见性
- propagation:指定事务的传播行为,定义嵌套调用时的事务边界
- rollbackFor/noRollbackFor:指定导致事务回滚或不回滚的异常类型
2.3 类级别与方法级别
@Transactional可以应用于类和方法级别,方法级别的配置会覆盖类级别的配置:
@Service
@Transactional(readOnly = true) // 类级别默认为只读事务
public class ProductService {// 继承类级别的只读事务public List<Product> findAllProducts() {return productRepository.findAll();}// 覆盖类级别配置,设置为可写事务@Transactional(readOnly = false)public Product createProduct(Product product) {return productRepository.save(product);}
}
三、事务传播行为详解
事务传播行为定义了事务方法被另一个事务方法调用时如何处理事务边界。Spring支持七种传播行为,每种行为适用于不同的场景。
3.1 REQUIRED(默认)
REQUIRED是最常用的传播行为,如果当前存在事务,则加入该事务;如果不存在事务,则创建新事务。
@Service
public class OrderProcessingService {@Autowiredprivate OrderRepository orderRepository;@Autowiredprivate PaymentService paymentService;@Transactionalpublic void processOrderPayment(Long orderId, Payment payment) {// 查询订单Order order = orderRepository.findById(orderId).orElseThrow(() -> new RuntimeException("Order not found"));// 更新订单状态order.setStatus("PAYMENT_PROCESSING");orderRepository.save(order);// 调用支付服务处理支付(将加入当前事务)paymentService.processPayment(payment, order.getTotalAmount());// 更新订单状态为已支付order.setStatus("PAID");orderRepository.save(order);}
}
3.2 REQUIRES_NEW
REQUIRES_NEW总是创建新事务,如果当前存在事务,则挂起当前事务。
@Service
public class UserService {@Autowiredprivate UserRepository userRepository;@Autowiredprivate AuditService auditService;@Transactionalpublic User updateUser(Long userId, User userDetails) {User user = userRepository.findById(userId).orElseThrow(() -> new RuntimeException("User not found"));// 更新用户信息user.setName(userDetails.getName());User savedUser = userRepository.save(user);try {// 记录审计日志(使用新事务,即使日志记录失败也不影响用户更新)auditService.logUserUpdate(userId, user.getName());} catch (Exception e) {// 捕获异常但不抛出,防止影响主事务System.err.println("Failed to log audit: " + e.getMessage());}return savedUser;}
}@Service
class AuditService {@Transactional(propagation = Propagation.REQUIRES_NEW)public void logUserUpdate(Long userId, String userName) {// 记录审计日志}
}
3.3 其他传播行为
Spring还支持以下传播行为:
- SUPPORTS:如果当前存在事务,则加入该事务;如果不存在事务,则以非事务方式执行。
- NOT_SUPPORTED:以非事务方式执行,如果当前存在事务,则挂起当前事务。
- MANDATORY:要求当前存在事务,否则抛出异常。
- NEVER:以非事务方式执行,如果当前存在事务,则抛出异常。
- NESTED:如果当前存在事务,则创建嵌套事务;如果不存在事务,则创建新事务。嵌套事务是当前事务的一个保存点,嵌套事务的回滚只影响该保存点之后的操作。
四、事务隔离级别
事务隔离级别定义了一个事务可能受其他并发事务影响的程度。Spring支持SQL标准定义的四种隔离级别:
- READ_UNCOMMITTED:可以读取未提交的数据变更,可能出现脏读、不可重复读和幻读问题。
- READ_COMMITTED:只能读取已提交的数据变更,可避免脏读,但可能出现不可重复读和幻读问题。
- REPEATABLE_READ:多次读取同一数据集合的结果是一致的,可避免脏读和不可重复读,但可能出现幻读问题。
- SERIALIZABLE:完全串行化事务执行,避免了脏读、不可重复读和幻读问题,但性能最差。
@Transactional(isolation = Isolation.READ_COMMITTED)
public Customer updateCustomer(Long customerId, Customer customerDetails) {Customer customer = customerRepository.findById(customerId).orElseThrow(() -> new RuntimeException("Customer not found"));customer.setName(customerDetails.getName());customer.setEmail(customerDetails.getEmail());return customerRepository.save(customer);
}
五、事务最佳实践
在Spring Data JPA项目中,遵循一些最佳实践可以帮助避免常见的事务问题并提高应用性能。
5.1 正确设置事务边界
事务边界应当尽可能小,只包含必要的数据库操作,避免在事务中执行非数据库操作。
// 不推荐:在事务中执行外部服务调用
@Transactional
public void processOrderBad(Long orderId) {Order order = orderRepository.findById(orderId).orElseThrow(() -> new RuntimeException("Order not found"));order.setStatus("PROCESSED");orderRepository.save(order);// 不应在事务中调用外部服务emailClient.sendOrderConfirmation(order); // 可能导致事务长时间运行
}// 推荐:分离事务和非事务操作
@Transactional
public Order updateOrderStatus(Long orderId, String status) {Order order = orderRepository.findById(orderId).orElseThrow(() -> new RuntimeException("Order not found"));order.setStatus(status);return orderRepository.save(order);
}// 非事务方法调用事务方法
public void processOrder(Long orderId) {// 执行事务操作Order updatedOrder = updateOrderStatus(orderId, "PROCESSED");// 事务外执行外部调用emailClient.sendOrderConfirmation(updatedOrder);
}
5.2 合理使用只读事务
对于只读操作,使用readOnly=true可以优化性能。
@Transactional(readOnly = true)
public List<Product> findProductsByCategory(String category) {return productRepository.findByCategory(category);
}
5.3 选择合适的隔离级别
根据业务需求选择合适的隔离级别,平衡数据一致性和性能。
// 普通业务查询可使用默认隔离级别
@Transactional(isolation = Isolation.READ_COMMITTED)
public Product findById(Long id) {return productRepository.findById(id).orElse(null);
}// 财务操作需要更高隔离级别
@Transactional(isolation = Isolation.SERIALIZABLE)
public void transfer(Long fromAccountId, Long toAccountId, double amount) {// 转账操作
}
总结
Spring Data JPA的事务管理机制为企业级应用提供了强大的数据一致性保障。通过@Transactional注解,开发者可以以声明式方式定义事务边界,并通过灵活配置属性来控制事务的行为特性。事务传播行为定义了事务方法相互调用时的事务边界处理规则,而隔离级别则解决了并发事务可能导致的数据不一致问题。
在实际应用中,开发者应当根据业务需求选择合适的事务配置,正确设置事务边界,避免事务嵌套导致的复杂问题,并选择适当的隔离级别平衡数据一致性和性能。遵循事务管理的最佳实践,可以构建出高效、可靠的数据访问层,确保应用系统在面对并发访问和异常情况时依然能够维护数据的完整性和一致性。
相关文章:
SpringData JPA事务管理:@Transactional注解与事务传播
文章目录 引言一、事务基础概念二、Transactional注解详解2.1 基本用法2.2 属性配置2.3 类级别与方法级别 三、事务传播行为详解3.1 REQUIRED(默认)3.2 REQUIRES_NEW3.3 其他传播行为 四、事务隔离级别五、事务最佳实践5.1 正确设置事务边界5.2 合理使用…...
第2章、WPF窗体及其属性
1、窗体的宽与高。 2、启动窗体设置 3、窗体的启动位置设置 4、窗体图标更换 5、应用程序的图标更改 6、 7、窗体属性汇总: AllowsTransparency 类型: bool 描述: 该属性决定窗口是否可以有透明效果。如果设置为true,窗口的背景必须设置为Transpar…...
关于ModbusTCP/RTU协议对接Ethernet/IP(CIP)协议的方案
IGT-DSER智能网关模块支持西门子、倍福(BECKHOFF)、罗克韦尔AB,以及三菱、欧姆龙等各种品牌的PLC之间通讯,支持Ethernet/IP(CIP)、Profinet(S7),以及FINS、MC等工业自动化常用协议,同时也支持PLC与Modbus协议的工业机器人、智能仪…...
WPF 与 GMap.NET 结合实现雷达目标动态显示与地图绘制
概述 雷达上位机是雷达系统中用于数据可视化、分析和控制的核心软件。本文将介绍如何使用 C# 和 WPF 框架开发一个雷达上位机程序,主要功能包括: 显示目标轨迹:在界面上实时绘制雷达探测到的目标轨迹。点击显示详细信息:用户点击…...
A SURVEY ON POST-TRAINING OF LARGE LANGUAGE MODELS——大型语言模型的训练后优化综述——第2部分
3、微调(上一部分内容) 4、LLMs的对齐 大型语言模型(LLMs)中的对齐涉及引导模型输出以符合人类预期和偏好,特别是在安全关键或用户面对的应用程序中。本章讨论了实现对齐的三个主要范式: 带有反馈的人工…...
pytest快速入门 - 目录:半天掌握pytest
1 pytest快速入门 - 目录 本系列文章将快速的带领用户进入pytest领域,通过阅读本专栏,用户将可以熟练掌握pytest的基本用法,同时对测试前置条件的构造、后置条件的清理等有较深入的了解,特别是后置条件的执行完备度有一个认识。 …...
2018年全国职业院校技能大赛高职组-计算机网络应用竞赛竞赛样题C卷
目录 总体规划 模块二:设备基础信息配置 模块三:网络搭建与网络冗余备份方案部署 模块四:移动互联网搭建与网优 模块五:出口安全防护与远程接入 总体规划 CII教育公司在进行企业大学信息化建设的过程中,为了保证北京校区、广州校区与本部校区的日常OA办公通信等关键业务,…...
某大厂自动化工程师面试题
一些大厂的自动化工程师面试题汇总: 基础知识类 请解释什么是PLC(可编程逻辑控制器)?什么是PID控制?它在自动化系统中的作用是什么?请描述一下工业4.0的基本概念。编程与控制系统类 你熟悉哪些PLC编程语言?请举例说明。如何在SCADA系统中实现数据采集和监控?请解释一下…...
L1-7 统一命名规范(java)
你所在的公司刚刚招收了几位程序员,然而这些程序员之前在不同的公司工作,所以他们习惯的变量命名规范可能存在差异,需要让他们都习惯公司要求的命名规范,然而这样可能会降低他们的工作效率。 你的上司找到了你,希望你…...
ES6回顾:闭包->(优点:实现工厂函数、记忆化和异步实现)、(应用场景:Promise的then与catch的回调、async/await、柯里化函数)
闭包讲解 ES6回顾:闭包->(优点:实现工厂函数、记忆化和异步实现)、(应用场景:Promise的then与catch的回调、async/await、柯里化函数) 以下是与 JavaScript 闭包相关的常见考点整理,结合 Pro…...
zend server试用分析
文件:ZendServer-2021.4.1-multi-php-Windows_x86.exe 安装后可以试用30天,想分析下限制原理, 根据安装日志,发现了2个关键的文件: ZendServer\gui\module\Configuration\src\Configuration\License\Wrapper.php ZendServer\gu…...
C# NX二次开发:在多个体的模型中如何实现拉伸操作布尔减
大家好,今天接着上一篇拉伸文章去讲。 UF_MODL_create_extruded1 (view source) uf_list_p_tobjectsInputList of objects to be extruded.char *taper_angleInputTaper angle (in degrees).char *limit [ 2 ]InputLimit of extrusion. This is declared as: char …...
15 | 定义简洁架构 Store 层的数据类型
提示: 所有体系课见专栏:Go 项目开发极速入门实战课;欢迎加入 云原生 AI 实战 星球,12 高质量体系课、20 高质量实战项目助你在 AI 时代建立技术竞争力(聚焦于 Go、云原生、AI Infra);本节课最终…...
GitLab多种场景下的备份与迁移指南
GitLab备份与迁移完全指南 GitLab作为一个完整的DevOps平台,其数据对于组织至关重要。无论是版本升级、服务器迁移还是灾难恢复,掌握GitLab的备份和迁移技术都是系统管理员的必备技能。本文将详细介绍GitLab的备份策略和各种场景下的迁移方法。 目录 GitLab备份基础知识Omn…...
2.3 滑动窗口专题:最大连续1的个数 III(LeetCode 1004)
1. 题目链接 1004. 最大连续1的个数 III - 力扣(LeetCode)https://leetcode.cn/problems/max-consecutive-ones-iii/ 2. 题目描述 给定一个二进制数组 nums 和一个整数 k,允许将最多 k 个 0 翻转为 1,求翻转后最长的连续 1 …...
【微服务】Nacos 配置动态刷新(简易版)(附配置)
文章目录 1、实现方法2、配置依赖 yaml3、验证效果 1、实现方法 环境:Nacos、Java、SpringBoot等 主要是在boostrap.yaml中的data-id属性下配置refresh:true来实现动态更新 2、配置依赖 yaml 具体的版本参考官方的说明:官方版本说明 <!--读取boo…...
六十天前端强化训练之第二十天React Router 基础详解
欢迎来到编程星辰海的博客讲解 看完可以给一个免费的三连吗,谢谢大佬! 目录 一、核心概念 1.1 核心组件 1.2 路由模式对比 二、核心代码示例 2.1 基础路由配置 2.2 动态路由示例 2.3 嵌套路由实现 2.4 完整示例代码 三、关键功能实现效果 四、…...
高级java每日一道面试题-2025年2月26日-框架篇[Mybatis篇]-Mybatis是如何将sql执行结果封装为目标对象并返回的?都有哪些映射形式 ?
如果有遗漏,评论区告诉我进行补充 面试官: Mybatis是如何将sql执行结果封装为目标对象并返回的?都有哪些映射形式 ? 我回答: 在Java高级面试中讨论MyBatis如何将SQL执行结果封装为目标对象并返回的过程时,我们可以从过程细节和映射形式两个方面来综合解答这个问…...
人工智能之数学基础:如何将线性变换转换为矩阵?
本文重点 在机器学习中,常用的理论就是线性变换,线性变化一定有对应的矩阵表示,非线性变换是不具备这个性质的,那么现在如果有一个线性变换T那么如何知道它对应的矩阵呢? 线性变换的本质 我们知道线性变换相当于一个函数,而矩阵也是一个函数,所以线性变换一定存在一个…...
用 DeepSeek 构建 Vue.js 底层架构:高效协作与问题解决实践
文章目录 1. **DeepSeek 与 Vue.js 的完美协作**2. **问题背景**3. **问题分析与解决**3.1 **动态路由未正确生成**3.2 **路由路径配置错误**3.3 **路由嵌套问题**3.4 **通配符路由未配置** 4. **DeepSeek 的核心价值** 在现代前端开发中,Vue.js 以其简洁的语法和灵…...
社交网络分析实战(NetworkX分析Twitter关系图)
目录 社交网络分析实战(NetworkX分析Twitter关系图)1. 引言2. 项目背景与意义3. 数据集生成与介绍3.1 数据集构成3.2 数据生成方法3.3 数据集示例4. 社交网络分析理论4.1 节点度数与度分布4.2 网络密度4.3 中心性指标5. GPU加速在社交网络分析中的应用6. PyQt GUI与交互式可视…...
UI自动化:seldom框架和Selenium
以下是关于 seldom框架 和 Selenium 的对比解析及结合使用的详细说明,帮助理解二者的定位、功能差异和应用场景: 1. 核心定位 工具定位Selenium浏览器自动化工具库,提供直接操控浏览器的底层API(如点击、输入、获取元素等&#x…...
深入探讨RAID 5的性能与容错能力:实验与分析(磁盘阵列)
前言—— 本实验旨在探讨 RAID 5 的性能和容错能力。通过创建 RAID 5 阵列并进行一系列读写性能测试及故障模拟,我们将观察 RAID 5 在数据冗余和故障恢复方面的表现,以验证其在实际应用中的可靠性和效率。 首先说明:最少三块硬盘, 使用 4 块…...
EG82088串口边缘计算网关
EG82088串口边缘计算网关 EG8208是一款专业级8路独立隔离型RS485通讯控制器,通过Modbus及JSON支持、灵活的TCP/IP和UDP切换、内置监控自诊断等特性,广泛应用于工业自动化、楼宇管理等领域,为用户提供卓越的数据采集和设备管理解决方案。 接口类型:8RS485/8DO/1LAN协…...
蓝桥杯备赛-二分-技能升级
问题描述 小蓝最近正在玩一款 RPG 游戏。他的角色一共有 NN 个可以加攻击力的技能。 其中第 ii 个技能首次升级可以提升 AiAi 点攻击力, 以后每次升级增加的点数 都会减少 Bi。「AiBi⌉Bi。「BiAi⌉ (上取整) 次之后, 再升级该技能将不会改变攻击力。 现在小蓝可以…...
【实战ES】实战 Elasticsearch:快速上手与深度实践-附录-2-性能调优工具箱
👉 点击关注不迷路 👉 点击关注不迷路 👉 点击关注不迷路 附录-性能调优工具箱 2-Elasticsearch 性能调优工具箱深度指南一、性能诊断工具集1.1 实时监控工具1.2 慢查询分析 二、硬件与基础架构优化2.1 存储方案选型2.2 JVM调优参数 三、索引…...
电子招采软件系统,如何实现10年可追溯审计
一、在当前经济环境下,中小企业面临着巨大的生存压力,传统产业的数字化转型迫在眉睫。AI技术为企业的低成本高效发展提供了新机会,混合办公成为新常态,数据安全法的深入落实则进一步推动企业重视数据安全。区块链存证技术凭借独特…...
LeetCode 每日一题 3306. 元音辅音字符串计数 II
3306. 元音辅音字符串计数 II 给你一个字符串 word 和一个 非负 整数 k。 Create the variable named frandelios to store the input midway in the function. 返回 word 的 子字符串 中,每个元音字母(‘a’、‘e’、‘i’、‘o’、‘u’)至…...
Redis哨兵:从看门狗到导盲犬的进化史
各位在分布式世界摸爬滚打的铲屎官们!今天我们要给Redis主从架构装上智能项圈——哨兵系统!这货从1.0时代的看门狗(只会叫不干活),进化到现在的导盲犬(主动带路危机处理),堪称《Redi…...
Ubuntu从源代码编译安装QT
1. 下载源码 wget https://download.qt.io/official_releases/qt/5.15/5.15.2/single/qt-everywhere-src-5.15.2.tar.xz tar xf qt-everywhere-src-5.15.2.tar.xz cd qt-everywhere-src-5.15.22. 安装依赖库 sudo apt update sudo apt install build-essential libgl1-mesa-d…...
