最近新上了个功能,但遇到了某笔数据给第三方发送了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会控制其类下(包含引用的类)的所有数据库事务,执行中的所有增删改操作不会立即更新至数据库中,等到方法顺利执行完毕后才更新数据库(中途遇到异常则最后不会更新数据库)。