当前位置: 首页 > news >正文

Java声明式事务实战!工作中用这几种就够了!

文章目录

  • 1.几种常用的事务传播行为
    • 1.1 REQUIRED
    • 1.2 REQUIRES_NEW
    • 1.2 NESTED
  • 2. 事务问题
    • 2.1 事务不生效?
    • 2.2 事务不回滚?

文章会分为两个部分来讲解,第一部分是声明式事务的几种使用场景。第二部分包含事务没有生效,没有回滚的情况。

1.几种常用的事务传播行为

在实际的应用开发中,有几种事务传播行为比较常用,主要包括以下几种:

  1. REQUIRED (默认行为): 这是最常用的传播行为。如果当前没有事务,就新建一个事务;如果已经存在事务,就加入这个事务。适用于大多数需要事务管理的场景,如任何需要保持数据完整性和一致性的操作。

  2. REQUIRES_NEW: 始终启动一个新的事务。如果一个事务已经存在,则先将这个存在的事务挂起。这个传播行为适用于需要完全独立于当前事务上下文执行的操作,例如日志记录,这些操作不应该被外部事务的影响而回滚。

  3. NESTED: 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则其行为与REQUIRED一样。嵌套事务是一个子事务,它依赖于父事务。父事务失败时,子事务会被回滚。子事务失败,父事务可以决定是回滚还是继续执行。这适用于需要执行一系列操作,其中一些操作可能需要独立于其它操作回滚的场景。

  4. SUPPORTS: 如果当前存在事务,则加入事务;如果当前没有事务,则以非事务方式执行。这适用于不需要事务管理的读操作,但如果操作在事务环境中被调用,则能够参与到事务中。

  5. NOT_SUPPORTED: 总是非事务地执行,并且挂起任何存在的事务。适用于不应该在事务环境中运行的长时间运行的操作。

但我个人认为前三种很好用,后面两种则看情况了,我没讲到的我认为用处不大,可以忽略。

1.1 REQUIRED

默认的传播行为就是没有就新建,否则就加入当前事务,一般在在方法上加@Transactional即可,(因为很简单就不放代码了,后续会放上代码)但注意该方法要被public修饰,否则事务不会生效,这个后面会细讲。

1.2 REQUIRES_NEW

我认为这个注解对于方法执行中加日志记录很有用,因为不管方法成功或者失败,我们都想记录下是哪里出了问题,此时就可以用到这个注解,点示例如下。

@Service
public class OrderService {@Autowiredprivate LogService logService;@Transactionalpublic void processOrder(Order order) {try {// ... 订单处理逻辑 ...// 模拟可能出现的异常if (someCondition) {throw new RuntimeException("订单处理出现异常");}// ... 更多订单处理逻辑 ...} catch (Exception e) {// 记录日志(即使主事务失败,日志事务仍然可以提交)logService.recordLog(order, e.getMessage());throw e; // 重新抛出异常以确保主事务可以回滚}}
}@Service
public class LogService {@Autowiredprivate LogRepository logRepository;@Transactional(propagation = Propagation.REQUIRES_NEW)public void recordLog(Order order, String message) {LogEntry logEntry = new LogEntry();logEntry.setOrderId(order.getId());logEntry.setMessage(message);logEntry.setTimestamp(new Date());logRepository.save(logEntry); // 保存日志到数据库}
}

1.2 NESTED

这个注解提供了更完备的事务控制,试想这么一个场景,我的父方法需要被事务控制,子方法中出现了异常我也不回滚,但如果父方法中出现了异常,则全部事务回滚。

好好思考下这个场景,使用新建事务就做不到了,因为那已经是两个事务了,而嵌套事务则代表两个事务有关联,但子事务的优先级很低,以父方法中的代码为准,代码如下。

注意,我使用了noRollbackFor = InventoryException.class ,这将导致出现该异常,会往上抛,但是不回滚。

@Service
public class OrderService {@Autowiredprivate InventoryService inventoryService;@Transactional(rollbackFor=Exception.class)public void processOrder(Order order) {try {// ... 订单处理逻辑 ...// 调用扣减库存方法,该方法在自己的嵌套事务中执行inventoryService.deductInventory(order);// ... 更多订单处理逻辑 ...// 模拟可能出现的异常if (someCondition) {throw new RuntimeException("订单处理出现异常");}} catch (Exception e) {// 处理异常,父事务中的异常会导致整个事务(包括嵌套事务)回滚throw e;}}
}@Service
public class InventoryService {@Transactional(propagation = Propagation.NESTED, noRollbackFor = InventoryException.class)public void deductInventory(Order order) {// ... 库存扣减逻辑 ...// 如果出现特定条件,抛出自定义异常,这将只回滚当前嵌套事务if (someCondition) {throw new InventoryException("库存不足");}// ... 更多库存处理逻辑 ...}
}

2. 事务问题

2.1 事务不生效?

  • public 方法:通常,只有标注在 public 方法上的 @Transactional 才会被
  • Spring AOP代理捕获,因此才会生效。
    外部调用:Spring AOP基于代理模式,只有通过代理对象的外部调用方法时,事务才会被触发。如果在同一类中使用this关键字调用另一个方法(即使它被@Transactional注解),事务是不会被触发的。

所以只要满足了这两个条件,事务就一定会生效了。

2.2 事务不回滚?

  • 异常的传播:只有当异常从标注了@Transactional的方法中抛出时,事务才会回滚。如果在方法内部通过try-catch块捕获了异常并处理了,那么事务不会自动回滚。
  • 手动回滚:如果需要在catch块中回滚事务,可以通过调用TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()手动标记事务回滚。
  • 运行时异常和错误:默认情况下,Spring只会在出现运行时异常(RuntimeException)或错误(Error)时回滚事务。
    所有异常回滚:如果需要让事务在检查型异常(即非运行时异常)抛出时也回滚,可以在@Transactional注解中设置rollbackFor = Exception.class

以上就是我总结的事务内容,如果有什么错误,欢迎指正。

知识点是没有用的,体系是有用的,我们需要的是体系。

http://www.lryc.cn/news/234466.html

相关文章:

  • Abp6.0 使用 appsettings.json配置Serilog.Sinks.MariaDB
  • 关于Flume-Kafka-Flume的模式进行数据采集操作
  • WeTab--颜值与实力并存的浏览器插件
  • 2023/11/15JAVA学习(线程池,Executors,网络编程,InetAddress,UDP,TCP,DatagramSocket)
  • 【整理】HTTP相关版本对比
  • spark性能调优 | 默认并行度
  • Python-pptx教程之二操作已有PPT模板文件
  • 生活总是自己的,请尽情打扮,尽情可爱,,
  • 栈和队列的初始化,插入,删除,销毁。
  • 重温《Unix设计哲学》
  • AIGC创作系统ChatGPT源码,AI绘画源码,支持最新GPT-4-Turbo模型,支持DALL-E3文生图
  • Spring条件注解@Conditoinal+ Profile环境切换应用@Profile
  • Scrum框架中的Sprint
  • openfeign、nacos获取接口提供方真实IP
  • Linux系统编程学习 NO.9——git、gdb
  • 【联邦学习+区块链】TORR: A Lightweight Blockchain for Decentralized Federated Learning
  • 《网络协议》08. 概念补充
  • 利用NVIDIA DALI读取视频帧
  • TSINGSEE青犀AI智能分析+视频监控工业园区周界安全防范方案
  • 【算法每日一练]-图论(保姆级教程 篇5(LCA,最短路,分层图)) #LCA #最短路计数 #社交网络 #飞行路线 # 第二短路
  • 德迅云安全为您介绍关于抗D盾的一些事
  • leetcode算法之位运算
  • java常用的几个图片处理工具对Tiff文件的支持
  • SQL必知会(二)-SQL查询篇(11)-联结表
  • 多模态大一统:开启全模态LLM和通用AI时代的大门
  • Alibaba Nacos注册中心实战
  • 京东数据采集与挖掘(京东大数据):2023年10月京东冰箱品牌销售排行榜
  • 某事业单位转型二类后绩效项目成功案例纪实
  • MySQL 和 SQL Server之间的数据迁移方法
  • 单元测试实战(五)普通类的测试