前提
一般我们在开发时,在方法或者类上加了@Transactional事务注解,然后会用 try catch 将可能会出问题的代码块包起来,在catch里面处理捕获的异常,但是,如果在catch里面没有把异常抛出去,此时事务是不会自动回滚的
异常模拟
比如这种情况:
这里既没有抛出异常,也没有手动回滚,在插入流水表之后,模拟报错,下面更新总支出和总缴存不会执行,但上面的流水已经插入了,看结果
这种情况显然不可取
手动回滚
在catch里面将手动回滚代码加上
//手动回滚
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
效果:
可见数据库并没有remark为:测试手动回滚 的流水数据,说明回滚生效
抛出异常实现自动回滚
将手动回滚注释,加上抛出异常
这里可以看到,新的测试自动回滚的流水也没有插入到流水表,抛出异常后已自动回滚
总结
在日常开发中,如果我们将多条操作数据库的代码用try catch 包起来后,在前面操作数据库代码执行成功,后面操作数据库代码执行失败的情况下,不会滚数据会出问题,常用的解决的方法就是
- 手动回滚事务
//手动回滚
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
- 抛出异常,交给springAOP处理
知识点:spring aop 异常捕获原理:被拦截的方法需显式抛出异常,并不能经任何处理,这样aop代理才能捕获到方法的异常,才能进行回滚,默认情况下aop只捕获runtimeException的异常,或通过配置来捕获特定的异常并回滚。