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



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

    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 {
    
        @Autowired
        private LogService logService;
    
        @Transactional
        public void processOrder(Order order) {
            try {
                // ... 订单处理逻辑 ...
    
                // 模拟可能出现的异常
                if (someCondition) {
                    throw new RuntimeException("订单处理出现异常");
                }
    
                // ... 更多订单处理逻辑 ...
            } catch (Exception e) {
                // 记录日志(即使主事务失败,日志事务仍然可以提交)
                logService.recordLog(order, e.getMessage());
                throw e; // 重新抛出异常以确保主事务可以回滚
            }
        }
    }
    
    @Service
    public class LogService {
    
        @Autowired
        private 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
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42

    1.2 NESTED

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

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

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

    @Service
    public class OrderService {
    
        @Autowired
        private 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("库存不足");
            }
    
            // ... 更多库存处理逻辑 ...
        }
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44

    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

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

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

  • 相关阅读:
    Java 爬虫 jvppeteer
    M2 Mac Xcode编译报错 ‘***.framework/‘ for architecture arm64
    优先队列题目:拼车
    MySQL数据库——SQL语言
    职业教育数字化成行业发展新方向
    Log4j “史诗级 ”漏洞背后:项目只有三位赞助者;RISC-V 基金会加速设计 RISC-V GPU;Linux 5.16 将延期发布 | 开源日报
    Android结构的介绍和资源使用
    接口测试总结分享(http与rpc)
    在EVE-NG 平台导入锐捷模拟器镜像流程
    使用U盘安装openSUSE-Leap-15.4-DVD-x86_64
  • 原文地址:https://blog.csdn.net/weixin_44353507/article/details/134480943