spring中多线程的事务部分与全部回滚思路。

本文探讨了Spring中多线程环境下事务的处理。当一个线程中的业务出错时,通常仅该线程的事务会回滚,而不影响其他线程。为了实现全部回滚,需要将线程内的业务抽取为独立类并让Spring管理。通过单元测试发现,即使线程内事务回滚,仍可能出现部分操作未回滚的情况。文章最后提出需求,期望在多线程中一个线程出错时,所有事务能全部回滚。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

spring事务是根据内部的ThreadLocal来启用的,一条线程代表一条单独的事务;在一个事务代码块内,启动多条线程,如果其中一条线程出错,其他线程并不会受影响,也不会触发spring的事务回滚。

  1. 示例代码:
    @Transactional(rollbackFor = Exception.class, propagation = Propagation.REQUIRED)
    public void add() {
        mapper.insert(new User(2L,"b"));
        Thread thread = new Thread(() -> {
             mapper.insert(new User(1L,"a"));
        });
        thread.start();
    }

以上代码如果在thread里面出错,并不会影响mapper.insert(new User(2L,"b"));的事务正常插入

mapper.insert(new User(1L,"a"));也会插入,这可能在某些业务就不符合业务逻辑了(有时出错了就得部分回滚或全部回滚)

1.先介绍如何让线程内的业务代码回滚,主线程的业务不回滚的情况。

想让线程内的事务让spring管理,就得把里面的业务归回spring管理,看代码吧,说的奇奇怪怪的。

  1. 把要在手动创建的线程中执行的业务抽出来为一个单独
### Java Spring多线程环境下使用线程池实现事务回滚的方法 在Java Spring框架中,当涉及到多线程操作时,确保事务的一致性和原子性变得尤为重要。特别是在使用线程池的情况下,由于每个线程可能独立于主线程运行,因此需要特别注意事务管理。 #### 1. 配置线程池并启用异步支持 为了使Spring能够管理和控制由线程池启动的任务中的事务,必须先配置线程池,并通过`@EnableAsync`注解开启异步支持: ```java @Configuration @EnableAsync public class AsyncConfig implements AsyncConfigurer { @Override public Executor getAsyncExecutor() { ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); executor.setCorePoolSize(7); executor.setMaxPoolSize(42); executor.setQueueCapacity(11); executor.initialize(); return executor; } } ``` 此段代码定义了一个自定义的线程池配置类[^2]。 #### 2. 使用 `Propagation.REQUIRES_NEW` 设置新的事务上下文 对于那些希望在线程内保持自己独立事务的行为来说,在其上加上`@Transactional(propagation=Propagation.REQUIRES_NEW)`是非常重要的。这会使得即使父级事务失败也不会影响到子事务的结果,反之亦然。 ```java @Service public class MyService { @Autowired private AnotherService anotherService; @Transactional public void mainMethod(){ try{ asyncMethod(); // 调用异步方法 } catch (Exception e){ throw new RuntimeException(e.getMessage()); } // 主业务逻辑... } @Async @Transactional(propagation = Propagation.REQUIRES_NEW) public Future<Boolean> asyncMethod() throws InterruptedException { boolean result = anotherService.doSomethingInNewTransaction(); if (!result) { throw new Exception("Error occurred during asynchronous processing"); } return new AsyncResult<>(true); } } ``` 上述代码展示了如何在一个服务类里既拥有同步又具备异步处理能力的同时还能维持良好的事务边界[^1]。 #### 3. 注意事项 - **异常传播**:默认情况下,只有未捕获的受检异常才会触发事务回滚。可以通过设置`rollbackFor`属性来指定哪些类型的异常应该引起回滚。 - **代理机制的影响**:因为AOP代理的原因,直接调用同一个对象内的其他带有`@Transactional`标记的方法不会按照预期工作。应始终让这些被标注的方法暴露给外部组件去调用。 - **资源竞争死锁风险**:多个并发执行的服务可能会尝试访问相同的数据库记录或其他共享资源,从而引发潜在的竞争条件或死锁情况。合理设计应用程序结构以及采用乐观锁定策略可以帮助缓解这些问题。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值