spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

作者:ratelfu 时间:2023-07-06 10:50:33 

本文是基于springboot完成测试测试代码地址如下:
https://github.com/Dr-Water/springboot-action/tree/master/springboot-shiro

一、 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;

/**
    * 测试@Transactional 注解加到service层事务是否回滚
    */
   @RequestMapping("/tx")
   public void serviceTX(){
       transactionalService.controllerTX();
   }
   /**
    * 测试@Transactional 注解加到Controller层事务是否回滚
    */
   @Transactional(rollbackFor = Exception.class)
   @RequestMapping("/ctx2")
   public void cTX2(){
       userDao.update();
       System.out.println(2/0);
       jwtUserDao.update();
   }

/**
    * 测试@Transactional 注解加到Controller层事务是否回滚
    * 这里在Controller层为了方便直接调用了dao层,在实际开发中dao层即可在Controller层调用也可以在service层调用,
    * 比如service层只是直接调用dao层一个方法,此外没有任何操作,那么这时候完全不用写service层的方法,直接在Controller调用dao层即可,
    * 当然如果公司有规范,必须严格按照mvc的模式进行开发,则另说
    */
   @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>

数据库原始数据:

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

浏览器中输入:http://localhost:8081/tx/tx,由于本次使用测试代码进行统一的异常处理所以浏览器的返回数据如下:

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

控制台输出如下:

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

查看数据库中的数据并没有被修改
浏览器中输入:http://localhost:8081/tx/ctx2,

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

查看数据库中的数据并没有被修改
由此可以得出 :@Transactional 加到Controller层也是生效的,但是为了规范起见,还是加到service层上。

问题二、 @Transactional 注解中用不用加rollbackFor = Exception.class 这个属性值

spring的api doc中有折磨一句描述:

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

红框中的内容如下:

rolling back on RuntimeException and Error but not on checked exceptions

大致意思就默认情况下,当程序发生 RuntimeException 和 Error 的这两种异常的时候事务会回滚,但是如果发生了checkedExcetions ,如fileNotfundException 则不会回滚,所以 rollbackFor = Exception.class 这个一定要加!
验证如下:
浏览器输入:http://localhost:8081/tx/ctx3
控制台输出如下:

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

这时候查看数据库中的数据并没有被修改
浏览器输入:http://localhost:8081/tx/ctx4

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

这时候查看数据库数据已经被修改:

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

spring声明式事务 @Transactional 不回滚的多种情况以及解决方案

问题三:事务调用嵌套问题具体结果如下代码:


/**
    * 同类中在方法a中调用b
    * a没有事务,b有 ,异常发生在b中 不会回滚
    */
   @RequestMapping("/a1")
   public void a1(){
       transactionalService.a1();
   }

/**
    * 同类中在方法a中调用b
    * a没有事务,b有 ,异常发生在a中 不会回滚
    */
   @RequestMapping("/a2")
   public void a2(){
       transactionalService.a2();
   }
   /**
    * 同类中在方法a中调用b
    * a有事务,b没有 ,异常发生在b中 会回滚
    */
   @RequestMapping("/a3")
   public void a3(){
       transactionalService.a3();
   }
   /**
    * 同类中在方法a中调用b
    * a有事务,b没有 ,异常发生在a中 会回滚
    */
   @RequestMapping("/a4")
   public void a4(){
       transactionalService.a4();
   }
   /**
    * 同类中在方法a中调用b
    * a有事务,b也有 ,异常发生在b中 会回滚
    */
   @RequestMapping("/a5")
   public void a5(){
       transactionalService.a5();
   }
   /**
    * 同类中在方法a中调用b
    * a有事务,b也有 ,异常发生在a中 会回滚
    */
   @RequestMapping("/a6")
   public void a6(){
       transactionalService.a6();
   }

/**
    *a类中调用b类中的方法
    * a中有事务,b中也有 会回滚
    *
    */
   @RequestMapping("/b5")
   public  void b5(){
       transactionalService.b5();
   }

/**
    *a类中调用b类中的方法
    * a中有事务,b中没有 会回滚
    *
    */
   @RequestMapping("/b6")
   public  void b6(){
       transactionalService.b6();
   }

/**
    *a类中调用b类中的方法
    * a没有事务,b中有 不会回滚
    *
    */
   @RequestMapping("/b7")
   public  void b7(){
       transactionalService.b7();
   }

/**
    *a类中调用b类中的方法
    * a没有事务,b中没有 不会回滚
    *
    */
   @RequestMapping("/b8")
   public  void b8(){
       transactionalService.b8();
   }

总结:如果在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

来源:https://blog.csdn.net/weter_drop/article/details/103582742

标签:spring,@Transactional,不回滚
0
投稿

猜你喜欢

  • Java反转链表测试过程介绍

    2022-10-02 20:10:03
  • Java安全之Tomcat6 Filter内存马问题

    2022-11-20 07:29:23
  • C# 获取硬件参数的实现方法

    2023-11-04 21:30:38
  • AQS(AbstractQueuedSynchronizer)抽象队列同步器及工作原理解析

    2023-02-24 22:20:09
  • Android入门之RelativeLayout、FrameLayout用法分析

    2021-10-16 03:21:32
  • java数据结构与算法之快速排序详解

    2023-02-23 10:23:43
  • Spring的refresh()方法相关异常解析

    2021-12-08 07:39:07
  • Java的Comparable,Comparator和Cloneable三大接口详解

    2023-07-14 11:18:48
  • C#中使用强制类型实现字符串和ASCII码之间的转换

    2022-05-15 14:57:48
  • MybatisPlus #{param}和${param}的用法详解

    2023-02-02 13:08:10
  • Java源码解析之接口List

    2022-06-13 08:46:44
  • Android编程实现左右滑动切换背景的方法

    2022-03-21 17:56:49
  • springboot 在idea中实现热部署的方法

    2022-10-24 22:27:19
  • Java基于Socket实现多人聊天室

    2022-11-08 14:11:12
  • Java优化if-else代码的实战记录

    2023-05-16 22:55:18
  • 基于ChatGPT+SpringBoot实现智能聊天AI机器人接口并上线至服务器的方法

    2023-07-01 06:19:34
  • SpringMVC拦截器配置及运行流程解析

    2023-03-30 15:38:57
  • java 微信随机红包算法代码实例

    2022-12-03 12:29:08
  • Java 详解循环屏障CyclicBarrier如何实现多线程分段等待执行完成

    2022-02-16 04:50:49
  • android自定义环形对比图效果

    2023-06-15 19:49:19
  • asp之家 软件编程 m.aspxhome.com