SpringBoot多线程Transactional抛出异常会回滚吗?

@Transactional

@Transactional 是一个常用的 Java 注解,通常在框架中使用,如 Spring,用于对方法或类实现事务管理。当一个方法或类被 @Transactional 注解标记时,表示其中的代码应该在一个事务中执行。
事务是一个逻辑工作单元,它包含一个或多个数据库操作。通过应用 @Transactional 注解,确保在方法内部的所有数据库操作要么全部成功,要么如果有任何一个失败,则所有操作都会被回滚,使得数据库恢复到原始状态。
以下是 @Transactional 的工作原理:

1.如果一个方法被标记为 @Transactional,则在方法执行之前会启动一个事务。
2.如果方法成功执行(即没有抛出运行时异常),事务将被提交,对数据库的任何更改都会被保存。
3.如果在方法执行期间发生异常,事务将被回滚,并且在该事务中对数据库所做的任何更改都将被丢弃。
4.如果被注解的方法调用其他同样被 @Transactional 注解标记的方法,同一个事务会传播到嵌套的方法调用中。换句话说,所有方法将共享同一个事务。
5.@Transactional 注解可以应用于方法级别或类级别。当应用于类级别时,该类的所有公共方法都会自动在事务中执行。
6.还可以指定其他属性来自定义事务的行为,例如隔离级别、传播行为、只读模式和超时时间。

总的来说,@Transactional 是一个强大的工具,用于在 Java 应用程序中管理事务行为。它有助于确保数据的一致性,并使开发人员能够处理复杂的数据库操作,而无需担心手动事务管理。

多线程异常处理

现在思考一个小问题,假如A是一个线程,a1、a2…是A的子线程,那么a1抛出异常的时候,A能捕捉到异常吗?

答案是不完全能:

线程设计的理念:“线程的问题应该线程自己本身来解决,而不要委托到外部。”,正常情况下,如果不做特殊的处理,在主线程中是不能够捕获到子线程中的异常的。但是还是可以通过注册uncaughtException捕获子线程中的异常(仅限RuntimeException)。

            Thread t = new Thread(new Runnable() {
                @Override
                public void run() {
                    save(new PsycheInfo());
                    log.warn(Thread.currentThread().getName() + "insert success");
                }
            });
            t.setUncaughtExceptionHandler(new Thread.UncaughtExceptionHandler() {
                @Override
                public void uncaughtException(Thread t, Throwable e) {
                    e.printStackTrace();
                }
            });

Spring实现事务的原理是通过ThreadLocal把数据库连接绑定到当前线程中,同一个事务中数据库操作使用同一个jdbc connection,新开启的线程获取不到当前jdbc connection。那么新创建的线程中的事务如何把任务添加到主线程的事务中去呢?

spirng的事务传播机制

  1. REQUIRED(默认):
    如果当前没有事务,就新建一个事务。如果已经存在一个事务中,加入到这个事务中。
  2. SUPPORTS:
    如果当前存在事务,就加入这个事务;如果当前没有事务,就以非事务方式执行。
  3. MANDATORY:
    该方法必须在一个事务中运行,如果当前没有事务,则抛出异常。
  4. REQUIRES_NEW:
    创建一个新事务,如果当前存在事务,将其挂起。
  5. NOT_SUPPORTED:
    以非事务方式执行操作,如果当前存在事务,将其挂起。
  6. NEVER:
    以非事务方式执行,如果当前存在事务,则抛出异常。

使用方式类似于:

@Transactional(propagation = Propagation.NOT_SUPPORTED)
public void transactionalMethod() {
    // Your business logic
}

合理的使用事务传播,就可以把新创建的事务加入到已有的事务中。

Spring事务的原理

一句话:Spring实现事务的原理是通过ThreadLocal把数据库连接绑定到当前线程中,同一个事务中数据库操作使用同一个jdbc connection,新开启的线程获取不到当前jdbc connection。

也就是说,我没法可以在新建的线程里获取到主线程使用的同一个Connection,如果我们能通过Connection关闭线程自动提交改成手动提交,我们就可以自己控制什么时候回滚事务。

我的实现方法(抛砖引玉)

    @Autowired
    private DataSource dataSource;
    @Autowired
    private ThreadPoolTaskExecutor threadPoolTaskExecutor;

    @Transactional(rollbackFor = Exception.class)
    @Override
    public void testAsync() throws InterruptedException {
        CountDownLatch countDownLatch = new CountDownLatch(10);
        CountDownLatch mainDownLatch = new CountDownLatch(1);
        AtomicBoolean atomicBoolean = new AtomicBoolean(false);
        Thread.UncaughtExceptionHandler uncaughtExceptionHandler = (t, e) -> {
            atomicBoolean.set(false);
            System.out.println(t.getName());
            e.printStackTrace();
        };
        for (int i = 0; i < 10; i++) {
            Thread t = new Thread(() -> {
                try {
                    Connection connection = dataSource.getConnection();
                    connection.setAutoCommit(false);
                    save(new PsycheInfo());
                    countDownLatch.countDown();
                    if (countDownLatch.getCount() == 2) {
                        throw new RuntimeException("test exception");
                    }
                    log.warn(Thread.currentThread().getName() + "insert success");
                    mainDownLatch.await();
                    if (!atomicBoolean.get()) {
                        connection.rollback();
                    }
                } catch (SQLException sqlException) {
                    throw new RuntimeException(sqlException.getMessage());
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }finally {
                    countDownLatch.countDown();
                }
            });
            t.setUncaughtExceptionHandler(uncaughtExceptionHandler);
            threadPoolTaskExecutor.submit(t);
        }
        countDownLatch.await();
        if (atomicBoolean.get()) {
            mainDownLatch.countDown();
        }else {
            System.out.println("test exception");
            mainDownLatch.countDown();
            throw new InterruptedException("test exception");
        }
    }

在这个实现方法中,我们需要设置两个CountDownLatch,countDownLatch是用来记录线程执行的状态,等十个线程执行完毕由主线程判断是否有必要回滚,atomicBoolean就是回滚的标志。子线程等待主线程的mainDownLatch,判断是否回滚。

  • 24
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值