Bootstrap

@Transactional注解对事务回滚机制详解

最近新上了个功能,但遇到了某笔数据给第三方发送了httpPost请求 但是自己表数据全部回滚没有插入记录的情况(送完httpPost后,代码遇出错抛出异常,触发了事务的回滚),加上之前对数据库事务一直不太清楚,趁这个机会一次弄懂。

spring的@Transactional详细讲解请看下面这篇文章。

透彻的掌握 Spring 中@transactional 的使用 - 知乎

这边主要讲解实际使用场景。

假如我现在有个test方法在@Repository类A下:

 public void test(){
        // 数据准备
        UpConfigDto upConfigDto = new UpConfigDto();
        upConfigDto.setUpId("test001");
        upConfigDto.setUpCode("test001");
        // 存入数据库
        upConfigRepository.save(upConfigDto);
        // 手动抛出异常
        throw new RuntimeException();
    }

下面有以下几个场景及结果:

1.类A没有@Transactional注解

结果:在执行完save方法后表中就新插入数据,抛出异常后表中数据不回滚

2.类A@Transactional注解

结果:在执行完save方法后表中没有新插入数据,抛出异常后表中数据不变

3.类A@Transactional注解,且注释手动抛出异常代码(即期望正常存数)

结果:在执行完save方法后表中没有新插入数据,方法走完后表中有新增数据

接下来让我们把场景变得稍微复杂一点,修改test方法,saveTestDataB存B表数据,saveTestDataC存C表数据,如下所示:

 public void test(){
        // B类存测试数据方法
        ClassB.saveTestDataB();
        // C类存测试数据方法
        ClassC.saveTestDataC();
        // 手动抛出异常
        throw new  RuntimeException();
    }

测试后得出以下结果:

4.类A@Transactional注解,classB和classC没有@Transactional注解

结果:在执行完saveTestDataB和saveTestDataC方法后表中都没有新插入数据,抛出异常后表中数据不变

5.类A@Transactional注解,且注释手动抛出异常代码(即期望正常存数)

结果:在执行完saveTestDataB和saveTestDataC方法后表中都没有新插入数据,方法走完后表中有新增数据

6.类A没有@Transactional注解,classB和classC没有@Transactional注解

结果:在执行完saveTestDataB方法后即对应数据,在执行完saveTestDataC方法后即对应数据

类A抛出异常后两表中数据不回滚。

7.类A没有@Transactional注解,classB@Transactional注解,classC没有@Transactional注解

结果:在执行完saveTestDataB方法后即对应数据,在执行完saveTestDataC方法后即对应数据

类A抛出异常后两表中数据不回滚。

等一下,第7个结果好像有点不对劲,但似乎又没什么不对劲。那补一个场景:

8.类A@Transactional注解,classB@Transactional注解,classC没有@Transactional注解

结果:在执行完saveTestDataB和saveTestDataC方法后表中都没有新插入数据,抛出异常后表中数据不变。

结论:@Transactional会控制其类下(包含引用的类)的所有数据库事务,执行中的所有增删改操作不会立即更新至数据库中,等到方法顺利执行完毕后才更新数据库(中途遇到异常则最后不会更新数据库)。

;