这篇文章给大家分享的是有关如何解决spring声明式事务@Transactional不回滚的多种情况问题的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。
一、 spring 事务原理
一、Spring事务原理
在使用JDBC事务操作数据库时,流程如下:
//获取连接 1.Connection con = DriverManager.getConnection()//开启事务2.con.setAutoCommit(true/false);3.执行CRUD//提交事务/回滚事务 4. con.commit() / con.rollback();//关闭连接5. conn.close();
Spring本身并不提供事务,而是对JDBC事务通过AOP做了封装,隐藏了2和4的操作,简化了JDBC的应用。
spring对JDBC事务的封装,是通过AOP动态代理来实现的,在调用目标方法(也就是第3步)前后会通过代理类来执行事务的开启、提交或者回滚操作。
spring事务使用的两个不可忽略点:
注意关键词 “动态代理”,这意味着要生成一个代理类,那么我们就不能在一个类内直接调用事务方法,否则无法代理,而且该事务方法必须是public,如果定义成 protected、private 或者默认可见性,则无法调用!
问题一、@Transactional 应该加到什么地方,如果加到Controller会回滚吗?
@Transactional 最好加到service层,加到Controller层也是生效的,但是为了规范起见,还是加到service层上。
下载代码并启动难项目进行验证:主要代码如下:
Controller层代码如下:
@Autowired private TransactionalService transactionalService; @Autowired private UserDao userDao; @Autowired private JwtUserDao jwtUserDao; @RequestMapping("/tx") public void serviceTX(){ transactionalService.controllerTX(); } @Transactional(rollbackFor = Exception.class) @RequestMapping("/ctx2") public void cTX2(){ userDao.update(); System.out.println(2/0); jwtUserDao.update(); } @Transactional(rollbackFor = Exception.class) @RequestMapping("/ctx2") public void cTX2(){ userDao.update(); //手动抛出一个RuntimeException System.out.println(2/0); jwtUserDao.update(); }
service层的主要代码
@Autowired private UserDao userDao; @Autowired private JwtUserDao jwtUserDao; @Transactional(rollbackFor = Exception.class) public void controllerTX(){ userDao.update(); //手动抛出一个RuntimeException System.out.println(2/0); jwtUserDao.update(); }
dao层sql语句如下:
<update id="update"> UPDATE jwt_user SET username ='wangwuupdate' WHERE user_id= 2 </update> <update id="update"> UPDATE user SET username ='zsupdate' WHERE id= 2 </update>
数据库原始数据:
浏览器中输入:http://localhost:8081/tx/tx,由于本次使用测试代码进行统一的异常处理所以浏览器的返回数据如下:
控制台输出如下:
查看数据库中的数据并没有被修改
浏览器中输入:http://localhost:8081/tx/ctx2,
查看数据库中的数据并没有被修改
由此可以得出 :@Transactional 加到Controller层也是生效的,但是为了规范起见,还是加到service层上。
问题二、 @Transactional 注解中用不用加rollbackFor = Exception.class 这个属性值
spring的api doc中有折磨一句描述:
红框中的内容如下:
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
这时候查看数据库数据已经被修改:
问题三:事务调用嵌套问题具体结果如下代码:
@RequestMapping("/a1") public void a1(){ transactionalService.a1(); } @RequestMapping("/a2") public void a2(){ transactionalService.a2(); } @RequestMapping("/a3") public void a3(){ transactionalService.a3(); } @RequestMapping("/a4") public void a4(){ transactionalService.a4(); } @RequestMapping("/a5") public void a5(){ transactionalService.a5(); } @RequestMapping("/a6") public void a6(){ transactionalService.a6(); } @RequestMapping("/b5") public void b5(){ transactionalService.b5(); } @RequestMapping("/b6") public void b6(){ transactionalService.b6(); } @RequestMapping("/b7") public void b7(){ transactionalService.b7(); } @RequestMapping("/b8") public void b8(){ transactionalService.b8(); }
如果在a方法中调用b方法不管是不是a和b是不是在同一个类中,只要a方法中没有事务,则发生异常的时候不会回滚,即:当a无事务时,则a和b均没有事务,当a有事务时,b如果有事务,则b事务会加到a事务中,二者为同一事务!
感谢各位的阅读!关于“如何解决spring声明式事务@Transactional不回滚的多种情况问题”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!