一 Transactional注解使用
@Tranasctional注解是Spring 框架提供的声明式注解事务解决方案,我们在开发中使用事务保证方法对数据库操作的原子性,要么全部成功,要么全部失败。
@Tranasctional注解基本使用
这段代码是一个简单的使用@Tranasctional注解,需要删除临时表中所有数据,对数据库进行了操作所以使用了该注解。同样的对于新增、修改操作也需要使用该注解,在查询操作则是加上加上参数@Transactional(readOnly = true)
在后端API编写过程中,大家可能在service层的涉及到数据库操作的方法中使用该注解,对于和我一样的新手大家可能更多的是被项目经理要求说在涉及到数据库操作的方法上必须使用@Transactional注解,也只是模模糊糊知道使用该注解作用是发生错误时让数据库回滚,但最近我在使用该注解过程中就遇到了一些坑,在这里简单记录下。
二 @Transactional与try catch同时使用
try catch大家想必也不陌生,用来处理异常,但大家在使用@Tranasctional注解的方法内又使用了try catch 来处理异常,大家有没有考虑过这样做对吗?大家可能是最初像我一样只求简单把功能实现,对单表操作好像没什么问题,也并没有去深究我应不应该这样写,所以这次我就遇到了坑!
这次任务是涉及到新增用户时在方法内对三张表进行存储操作,要求当任意一张表发生错误时三张表全部进行回滚,业务代码大致如下:
当发生错误时我在catch里面打印出了错误,但是在测试过程中发现了:当我第一、二张表存入成功,但第三张表存入失败后,按照设想应该三张表全部回滚,但事实只对第三张表进行了回滚,一二两个表还是存入了数据库。
查阅了原因是存在@Tranasctional注解失效的问题。
三 @Tranasctional注解失效
如若在添加数据的过程中,中间有一个数据添加失败,那么catch的作用就是捕获该异常, 只让这一条数据不添加到数据库中。 后面的数据或者是这条数据之前的数据,依然可以添加到数据库中。目的是不让业务中断 因为catch你已经捕获异常了,所以业务不会中断。但Transactional注解的作用是添加数据的过程中,中间有一个数据添加失败,那么之前添加到数据库中的数据全部回滚。后面的数据也不会添加到数据库中保证数据安全。
当在方法中异常被catch捕获到,catch对其进行了处理,在catch中即没有继续抛出异常也没有手动回滚数据库,便会出现:异常被catch捕获到做了处理,异常的生命周期已经结束,而并没有将异常抛出所以,所以没有被@Transactional注解捕获到,导致该注解失效,数据库没有全部回滚该事务。
四 解决方法
解决方法在上面也提到了,很简单,在catch中继续抛出异常或者手动回滚事务。两种方法我均展示在下面代码中。手动回滚和抛出异常选择一种其实就可以实现了。
以上内容仅是我对项目中遇到的问题做一些记录,水平有限,若有不对欢迎指正!