这是我参与11月更文挑战的第17天,活动详情查看:2021最后一次更文挑战
本文是基于springboot完成测试测试代码地址如下:
github.com/Dr-Water/sp…
一、 spring 事务原理
一、Spring事务原理
在使用JDBC事务操作数据库时,流程如下:
1 | java复制代码//获取连接 |
Spring本身并不提供事务,而是对JDBC事务通过AOP做了封装,隐藏了2和4的操作,简化了JDBC的应用。
spring对JDBC事务的封装,是通过AOP动态代理来实现的,在调用目标方法(也就是第3步)前后会通过代理类来执行事务的开启、提交或者回滚操作。
spring事务使用的两个不可忽略点:
- 注意关键词 “动态代理”,这意味着要生成一个代理类,那么我们就不能在一个类内直接调用事务方法,否则无法代理,
- 而且该事务方法必须是public,如果定义成 protected、private 或者默认可见性,则无法调用!
一@Transactional 应该加到什么地方,如果加到Controller会回滚吗?
- @Transactional 最好加到service层,加到Controller层也是生效的,但是为了规范起见,还是加到service层上。
- 下载代码并启动难项目进行验证:主要代码如下:
Controller层代码如下:
1 | java复制代码 @Autowired |
service层的主要代码
1 | java复制代码 @Autowired |
dao层sql语句如下:
1 | java复制代码 <update id="update"> |
数据库原始数据:
浏览器中输入:http://localhost:8081/tx/tx,由于本次使用测试代码进行统一的异常处理所以浏览器的返回数据如下:
控制台输出如下:
查看数据库中的数据并没有被修改
浏览器中输入:http://localhost:8081/tx/ctx2,
查看数据库中的数据并没有被修改
由此可以得出 :@Transactional 加到Controller层也是生效的,但是为了规范起见,还是加到service层上。
问题二、 @Transactional 注解中用不用加rollbackFor = Exception.class 这个属性值
spring的api doc中有折磨一句描述:
红框中的内容如下:
1 | java复制代码rolling back on RuntimeException and Error but not on checked exceptions |
==大致意思就默认情况下,当程序发生 RuntimeException 和 Error 的这两种异常的时候事务会回滚,但是如果发生了checkedExcetions ,如fileNotfundException 则不会回滚,所以 rollbackFor = Exception.class 这个一定要加!==
验证如下:
浏览器输入:http://localhost:8081/tx/ctx3
控制台输出如下:
这时候查看数据库中的数据并没有被修改
浏览器输入:http://localhost:8081/tx/ctx4
这时候查看数据库数据已经被修改:
问题三:事务调用嵌套问题具体结果如下代码:
1 | java复制代码/** |
总结:如果在a方法中调用b方法不管是不是a和b是不是在同一个类中,只要a方法中没有事务,则发生异常的时候不会回滚,即:当a无事务时,则a和b均没有事务,当a有事务时,b如果有事务,则b事务会加到a事务中,二者为同一事务!
四、总结
在springboot中默认是开启事务的,在service层的方法加上@Transactional(rollbackFor = Exception.class) 注解即可实现事务 如果在方法a中调用方法b 如果要实现事务,则只需要在方法上加上@Transactional(rollbackFor = Exception.class) 即可!
如果业务需要,一定要抛出checked异常的话,可以通过rollbackFor属性指定异常类型即可。有兴趣的可以动手验证一下,这里不再赘述。
五、 参考链接
Spring Boot中的事务管理
深入理解 Spring 之 SpringBoot 事务原理
声明式事务不回滚@Transactional的避坑正确使用
Spring声明式事务不回滚问题
spring 事务应用误区总结:那些导致事务不回滚的坑
你的Spring事务为什么不会自动回滚,包含异常的分类
Java异常之checked与unchecked
本文转载自: 掘金