解决@Transactional注解事务不回滚不起作用的问题_java

来源:脚本之家  责任编辑:小易  

这几天在项目里面发现我使用@Transactional注解事务之后,抛了异常居然不回滚。后来终于找到了原因。

如果你也出现了这种情况,可以从下面开始排查。

一、特性

先来了解一下@Transactional注解事务的特性吧,可以更好排查问题

1、service类标签(一般不建议在接口上)上添加@Transactional,可以将整个类纳入spring事务管理,在每个业务方法执行时都会开启一个事务,不过这些事务采用相同的管理方式。

2、@Transactional 注解只能应用到 public 可见度的方法上。 如果应用在protected、private或者 package可见度的方法上,也不会报错,不过事务设置不会起作用。

3、默认情况下,Spring会对unchecked异常进行事务回滚;如果是checked异常则不回滚。

辣么什么是checked异常,什么是unchecked异常

java里面将派生于Error或者RuntimeException(比如空指针,1/0)的异常称为unchecked异常,其他继承自java.lang.Exception得异常统称为Checked Exception,如IOException、TimeoutException等

辣么再通俗一点:

你写代码出现的空指针等异常,会被回滚,文件读写,网络出问题,spring就没法回滚了。然后我教大家怎么记这个,因为很多同学容易弄混,你写代码的时候有些IOException我们的编译器是能够检测到的,说以叫checked异常,你写代码的时候空指针等死检测不到的,所以叫unchecked异常。这样是不是好记一些啦

4、只读事务:

@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true)

只读标志只在事务启动时应用,否则即使配置也会被忽略。

启动事务会增加线程开销,数据库因共享读取而锁定(具体跟数据库类型和事务隔离级别有关)。通常情况下,仅是读取数据时,不必设置只读事务而增加额外的系统开销。

二:事务传播模式

Propagation枚举了多种事务传播模式,部分列举如下:

1、REQUIRED(默认模式):业务方法需要在一个容器里运行。如果方法运行时,已经处在一个事务中,那么加入到这个事务,否则自己新建一个新的事务。

2、NOT_SUPPORTED:声明方法不需要事务。如果方法没有关联到一个事务,容器不会为他开启事务,如果方法在一个事务中被调用,该事务会被挂起,调用结束后,原先的事务会恢复执行。

3、REQUIRESNEW:不管是否存在事务,该方法总汇为自己发起一个新的事务。如果方法已经运行在一个事务中,则原有事务挂起,新的事务被创建。

4、 MANDATORY:该方法只能在一个已经存在的事务中执行,业务方法不能发起自己的事务。如果在没有事务的环境下被调用,容器抛出例外。

5、SUPPORTS:该方法在某个事务范围内被调用,则方法成为该事务的一部分。如果方法在该事务范围外被调用,该方法就在没有事务的环境下执行。

6、NEVER:该方法绝对不能在事务范围内执行。如果在就抛例外。只有该方法没有关联到任何事务,才正常执行。

7、NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中。如果没有活动事务,则按REQUIRED属性执行。它使用了一个单独的事务,这个事务拥有多个可以回滚的保存点。内部事务的回滚不会对外部事务造成影响。它只对DataSourceTransactionManager事务管理器起效。

三:解决Transactional注解不回滚

1、检查你方法是不是public的

2、你的异常类型是不是unchecked异常

如果我想check异常也想回滚怎么办,注解上面写明异常类型即可

@Transactional(rollbackFor=Exception.class) 

类似的还有norollbackFor,自定义不回滚的异常

3、数据库引擎要支持事务,如果是MySQL,注意表要使用支持事务的引擎,比如innodb,如果是myisam,事务是不起作用的

4、是否开启了对注解的解析

<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>

5、spring是否扫描到你这个包,如下是扫描到org.test下面的包

<context:component-scan base-package="org.test" ></context:component-scan>

6、检查是不是同一个类中的方法调用(如a方法调用同一个类中的b方法)

7、异常是不是被你catch住了

以上,以后有再添加吧~

补充:@Transactional事务几点注意

这里面有几点需要大家留意:

A. 一个功能是否要事务,必须纳入设计、编码考虑。不能仅仅完成了基本功能就ok。

B. 如果加了事务,必须做好开发环境测试(测试环境也尽量触发异常、测试回滚),确保事务生效。

C. 以下列了事务使用过程的注意事项,请大家留意。

1. 不要在接口上声明@Transactional ,而要在具体类的方法上使用 @Transactional 注解,否则注解可能无效。

2.不要图省事,将@Transactional放置在类级的声明中,放在类声明,会使得所有方法都有事务。故@Transactional应该放在方法级别,不需要使用事务的方法,就不要放置事务,比如查询方法。否则对性能是有影响的。

3.使用了@Transactional的方法,对同一个类里面的方法调用, @Transactional无效。比如有一个类Test,它的一个方法A,A再调用Test本类的方法B(不管B是否public还是private),但A没有声明注解事务,而B有。则外部调用A之后,B的事务是不会起作用的。(经常在这里出错)

4.使用了@Transactional的方法,只能是public,@Transactional注解的方法都是被外部其他类调用才有效,故只能是public。道理和上面的有关联。故在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不会报错,但事务无效。

5.经过在ICORE-CLAIM中测试,效果如下:

A.抛出受查异常XXXException,事务会回滚。

B.抛出运行时异常NullPointerException,事务会回滚。

C.Quartz中,execute直接调用加了@Transactional方法,可以回滚;间接调用,不会回滚。(即上文3点提到的)

D.异步任务中,execute直接调用加了@Transactional方法,可以回滚;间接调用,不会回滚。(即上文3点提到的)

E.在action中加上@Transactional,不会回滚。切记不要在action中加上事务。

F.在service中加上@Transactional,如果是action直接调该方法,会回滚,如果是间接调,不会回滚。(即上文3提到的)

G.在service中的private加上@Transactional,事务不会回滚。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持真格学网。如有错误或未考虑完全的地方,望不吝赐教。

您可能感兴趣的文章:Java注解@Transactional事务类内调用不生效问题及解决办法springboot中使用@Transactional注解事物不生效的坑带有@Transactional和@Async的循环依赖问题的解决Spring声明式事务@Transactional知识点分享Spring @Transactional注解失效解决方案springboot中事务管理@Transactional的注意事项与使用场景

  • 本文相关:
  • jfinal+ajax传参 array 数组方法 获取request中数组操作
  • 详解java的位操作符
  • java-4nio之channel之间的数据传输方法
  • java的struts框架中的if/else标签使用详解
  • java base64encoder详细介绍及简单实例
  • mybatis3传递多个参数(multiple parameters)
  • 浅谈java向下转型的意义
  • spring中的singleton和prototype的实现
  • spring整合weblogic jms实例详解
  • eclipse快捷键使用小结
  • @Transactional不回滚,求大神解决
  • Spring 声明式事务@Transactional不起作用,为什么
  • 请问各位java高手,spring mvc 事务提交无法回滚的...
  • spring service层 transactional注解 为什么异常没...
  • java中事务不回滚!
  • Spring中为什么@Transactional注解打在Controller...
  • spring 事务注解 不回滚
  • Java 用uilayerexception抛出异常,但是事务没有回...
  • 为什么java里面要用@Transactional注解
  • 什么情况下transactional注解失效
  • 网站首页网页制作脚本下载服务器操作系统网站运营平面设计媒体动画电脑基础硬件教程网络安全c#教程vbvb.netc 语言java编程delphijavaandroidiosswiftscala易语言汇编语言其它相关首页java注解@transactional事务类内调用不生效问题及解决办法springboot中使用@transactional注解事物不生效的坑带有@transactional和@async的循环依赖问题的解决spring声明式事务@transactional知识点分享spring @transactional注解失效解决方案springboot中事务管理@transactional的注意事项与使用场景jfinal+ajax传参 array 数组方法 获取request中数组操作详解java的位操作符java-4nio之channel之间的数据传输方法java的struts框架中的if/else标签使用详解java base64encoder详细介绍及简单实例mybatis3传递多个参数(multiple parameters)浅谈java向下转型的意义spring中的singleton和prototype的实现spring整合weblogic jms实例详解eclipse快捷键使用小结intellij idea 2020最新激活码(亲java使double保留两位小数的多方intellij idea 2020.2.3永久破解java8 十大新特性详解idea2020.2.2激活码与intellij ijava.net.socketexception: connjava写入文件的几种方法分享2020.2 intellij idea激活与ideajava 十六进制与字符串的转换java环境变量的设置方法(图文教程idea 隐藏target,iml等不需要展示的文件(从java中调用matlab详细介绍mybatis动态插入list传入list参数的实例代java web应用程序实现基础的文件下载功能java数字签名算法dsa实例详解springboot基于sentinel在服务上实现接口java.lang.outofmemoryerror 错误整理及解java中指定时区的3种方法eclipse添加servlet模板过程代码详解hashmap和list遍历方法及如何遍历删除元素
    免责声明 - 关于我们 - 联系我们 - 广告联系 - 友情链接 - 帮助中心 - 频道导航
    Copyright © 2017 www.zgxue.com All Rights Reserved