传播方式为REQUIRED的子事务因事务完成导致不再执行commit操作

今天日常环境中出现了一个事务相关bug,具体问题是这样的:我们有一个事件监听服务,监听到的各种事件都在这个里面处理。现在我们有一个业务单元初始化的时候会同步过来一批客户信息,事件监听服务里处理这批数据并批量插入数据库中,我们业务代码调用了平台框架中提供的批量处理操作,该操作会循环处理每一条数据,因为我们每插入一条数据会相应操作好几张表单,所以每个循环都会开启一个事务,即一条处理失败不影响其他数据的插入。但是推到日常环境中去之后发现,一条插入失败,所有都失败了,这明显不符合业务需求。所以查找bug,最后终于找到问题所在,在此记录一下。

一、源码分析

                              

                                                                                             图(1)

                              

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Spring 中,可以使用 `@Transactional` 注解来开启事务。在多线程环境下,如果需要控制事务回滚,可以采用以下两种方式: 1. 在线程中捕获异常并手动回滚事务线程中进行数据库操作时,如果出现异常,可以手动回滚事务。具体实现步骤如下: - 在父线程中开启事务,将事务对象传递给线程。 - 在线程中进行数据库操作,如果出现异常则手动回滚事务。 - 在父线程中判断线程是否抛出异常,如果有异常则回滚事务。 ``` @Service public class UserService { @Autowired private PlatformTransactionManager transactionManager; @Transactional public void doSomethingInTransaction() { // 在父线程中开启事务 DefaultTransactionDefinition definition = new DefaultTransactionDefinition(); TransactionStatus status = transactionManager.getTransaction(definition); try { // 在线程中进行数据库操作 CompletableFuture<Void> future = CompletableFuture.runAsync(() -> { // 获取线程的事务对象 TransactionStatus childStatus = TransactionAspectSupport.currentTransactionStatus(); try { // 线程中进行数据库操作 // ... } catch (Exception e) { // 手动回滚线程的事务 transactionManager.rollback(childStatus); throw e; } }); // 等待线程执行完毕 future.get(); // 判断线程是否抛出异常,如果有异常则回滚事务 if (future.isCompletedExceptionally()) { throw new RuntimeException("线程执行出错"); } // 提交事务 transactionManager.commit(status); } catch (Exception e) { // 回滚事务 transactionManager.rollback(status); throw e; } } } ``` 2. 使用 Spring 的 `TransactionTemplate` 来控制事务回滚 Spring 提供了 `TransactionTemplate` 类来简化事务的编程。可以通过 `TransactionTemplate` 的 `execute()` 方法来执行数据库操作,如果出现异常则自动回滚事务。具体实现步骤如下: - 在父线程中使用 `TransactionTemplate` 开启事务。 - 在线程中使用 `TransactionTemplate` 执行数据库操作。 - 如果线程抛出异常,则 `TransactionTemplate` 会自动回滚事务。 ``` @Service public class UserService { @Autowired private PlatformTransactionManager transactionManager; public void doSomethingInTransaction() { // 使用 TransactionTemplate 开启事务 TransactionTemplate template = new TransactionTemplate(transactionManager); template.execute(new TransactionCallbackWithoutResult() { @Override protected void doInTransactionWithoutResult(TransactionStatus status) { // 在线程中使用 TransactionTemplate 执行数据库操作 CompletableFuture<Void> future = CompletableFuture.runAsync(() -> { template.execute(new TransactionCallbackWithoutResult() { @Override protected void doInTransactionWithoutResult(TransactionStatus childStatus) { // 线程中进行数据库操作 // ... } }); }); // 等待线程执行完毕 future.join(); } }); } } ``` 无论使用哪种方式,都需要注意事务的传播机制。默认情况下,事务的传播机制是 `Propagation.REQUIRED`,即如果当前没有事务,则开启一个新的事务;如果当前已经存在事务,则加入该事务中。如果需要修改传播机制,可以在 `@Transactional` 注解中进行设置。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值