作者:溪~源

blog.csdn.net/xuan_lu/article/details/107797505

实际项目开发中,如果涉及到多张表操作时,为了保证业务数据的一致性,大家一般都会采用事务机制;好多小伙伴可能只是简单了解一下,遇到事务失效的情况,便会无从下手,溪源此篇文章给大家整理了一下常见Spring事务失效的场景,希望开发过程尽量避免踩坑,造成时间精力的浪费。

溪源按照最基本的使用方式以及常见失效场景优先级整理,先简单介绍一下具体失效场景:

  • 注解@Transactional配置的方法非public权限修饰;

  • 注解@Transactional所在类非Spring容器管理的bean;

  • 注解@Transactional所在类中,注解修饰的方法被类内部方法调用;

  • 业务代码抛出异常类型非RuntimeException,事务失效;

  • 业务代码中存在异常时,使用try…catch…语句块捕获,而catch语句块没有throw new RuntimeExecption异常;(最难被排查到问题且容易忽略)

  • 注解@Transactional中Propagation属性值设置错误即Propagation.NOT_SUPPORTED(一般不会设置此种传播机制)

  • mysql关系型数据库,且存储引擎是MyISAM而非InnoDB,则事务会不起作用(基本开发中不会遇到);

下面基于以上场景,溪源给小伙伴们详细解释;

非public权限修饰

参考Spring官方文档介绍,摘要、译文如下:

When using proxies, you should apply the @Transactional annotation only to methods with public visibility. If you do annotate protected, private or package-visible methods with the @Transactional annotation, no error is raised, but the annotated method does not exhibit the configured transactional settings. Consider the use of AspectJ (see below) if you need to annotate non-public methods.

译文

使用代理时,您应该只将@Transactional注释应用于具有公共可见性的方法。如果使用@Transactional注释对受保护的、私有的或包可见的方法进行注释,则不会引发错误,但带注释的方法不会显示配置的事务设置。如果需要注释非公共方法,请考虑使用AspectJ(见下文)。

简言之:@Transactional 只能用于 public 的方法上,否则事务不会失效,如果要用在非 public 方法上,可以开启 AspectJ 代理模式。

目前,如果@Transactional注解作用在非public方法上,编译器也会给与明显的提示,如图:

非Spring容器管理的bean

基于这种失效场景,有工作经验的大佬基本上是不会存在这种错误的;@Service 注解注释,StudentServiceImpl 类则不会被Spring容器管理,因此即使方法被@Transactional注解修饰,事务也亦然不会生效。

简单举例如下:

/*** @Author:qxy*/
//@Service
public class StudentServiceImpl implements StudentService {@Autowiredprivate StudentMapper studentMapper;@Autowiredprivate ClassService classService;@Override@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)public void insertClassByException(StudentDo studentDo) throws CustomException {studentMapper.insertStudent(studentDo);throw new CustomException();}
}

注解修饰的方法被类内部方法调用

这种失效场景是我们日常开发中最常踩坑的地方;在类A里面有方法a 和方法b, 然后方法b上面用 @Transactional加了方法级别的事务,在方法a里面 调用了方法b, 方法b里面的事务不会生效。为什么会失效呢?:

其实原因很简单,Spring在扫描Bean的时候会自动为标注了@Transactional注解的类生成一个代理类(proxy),当有注解的方法被调用的时候,实际上是代理类调用的,代理类在调用之前会开启事务,执行事务的操作,但是同类中的方法互相调用,相当于this.B(),此时的B方法并非是代理类调用,而是直接通过原有的Bean直接调用,所以注解会失效。推荐:250期面试题汇总

@Service
public class ClassServiceImpl implements ClassService {@Autowiredprivate ClassMapper classMapper;public void insertClass(ClassDo classDo) throws CustomException {insertClassByException(classDo);}@Override@Transactional(propagation = Propagation.REQUIRED)public void insertClassByException(ClassDo classDo) throws CustomException {classMapper.insertClass(classDo);throw new RuntimeException();}
}//测试用例:
@Testpublic void insertInnerExceptionTest() throws CustomException {classDo.setClassId(2);classDo.setClassName("java_2");classDo.setClassNo("java_2");classService.insertClass(classDo);}

测试结果:

java.lang.RuntimeExceptionat com.qxy.common.service.impl.ClassServiceImpl.insertClassByException(ClassServiceImpl.java:34)at com.qxy.common.service.impl.ClassServiceImpl.insertClass(ClassServiceImpl.java:27)at com.qxy.common.service.impl.ClassServiceImpl$$FastClassBySpringCGLIB$$a1c03d8.invoke(<generated>)

虽然业务代码报错了,但是数据库中已经成功插入数据,事务并未生效;

解决方案

类内部使用其代理类调用事务方法:以上方法略作改动

public void insertClass(ClassDo classDo) throws CustomException {
//         insertClassByException(classDo);((ClassServiceImpl)AopContext.currentProxy()).insertClassByException(classDo);}
//测试用例:@Testpublic void insertInnerExceptionTest() throws CustomException {classDo.setClassId(3);classDo.setClassName("java_3");classDo.setClassNo("java_3");classService.insertClass(classDo);}

业务代码抛出异常,数据库未插入新数据,达到我们的目的,成功解决一个事务失效问题;

数据库数据未发生改变;

注意:一定要注意启动类上要添加@EnableAspectJAutoProxy(exposeProxy = true)注解,否则启动报错:

java.lang.IllegalStateException: Cannot find current proxy: Set 'exposeProxy' property on Advised to 'true' to make it available.at org.springframework.aop.framework.AopContext.currentProxy(AopContext.java:69)at com.qxy.common.service.impl.ClassServiceImpl.insertClass(ClassServiceImpl.java:28)

异常类型非RuntimeException

这种事务失效场景也是非常难排查问题的,如果没有深究源码实现,估计要花费一番功夫啦;推荐:250期面试题汇总

@Service
public class ClassServiceImpl implements ClassService {@Autowiredprivate ClassMapper classMapper;//    @Override
//    @Transactional(propagation = Propagation.NESTED, rollbackFor = Exception.class)public void insertClass(ClassDo classDo) throws Exception {
//        即使此处使用代理对象调用内部事务方法,数据依然未发生回滚,事务机制亦然失效((ClassServiceImpl)AopContext.currentProxy()).insertClassByException(classDo);}@Override@Transactional(propagation = Propagation.REQUIRED)public void insertClassByException(ClassDo classDo) throws Exception {classMapper.insertClass(classDo);//抛出非RuntimeException类型throw new Exception();}
//测试用例:@Testpublic void insertInnerExceptionTest() throws Exception {classDo.setClassId(3);classDo.setClassName("java_3");classDo.setClassNo("java_3");classService.insertClass(classDo);}
}

运行结果:

业务代码抛出异常,但是数据库发生更新操作;

java.lang.Exceptionat com.qxy.common.service.impl.ClassServiceImpl.insertClassByException(ClassServiceImpl.java:35)at com.qxy.common.service.impl.ClassServiceImpl$$FastClassBySpringCGLIB$$a1c03d8.invoke(<generated>)at org.springframework.cglib.proxy.MethodProxy.invoke(MethodProxy.java:204)

数据库依然插入数据,不是我们想要的结果啊,赶紧修改吧,产品经理来追啦~

解决方案:

@Transactional注解修饰的方法,加上rollbackfor属性值,指定回滚异常类型:@Transactional(propagation = Propagation.REQUIRED,rollbackFor = Exception.class)

@Override@Transactional(propagation = Propagation.REQUIRED,rollbackFor = Exception.class)public void insertClassByException(ClassDo classDo) throws Exception {classMapper.insertClass(classDo);throw new Exception();}

捕获异常后,却未抛出异常

在事务方法中使用try-catch,导致异常无法抛出,自然会导致事务失效。

@Service
public class ClassServiceImpl implements ClassService {@Autowiredprivate ClassMapper classMapper;//    @Overridepublic void insertClass(ClassDo classDo) {((ClassServiceImpl)AopContext.currentProxy()).insertClassByException(classDo);}@Override@Transactional(propagation = Propagation.REQUIRED,rollbackFor = Exception.class)public void insertClassByException(ClassDo classDo) {classMapper.insertClass(classDo);try {int i = 1 / 0;} catch (Exception e) {e.printStackTrace();}}
}// 测试用例:@Testpublic void insertInnerExceptionTest() {classDo.setClassId(4);classDo.setClassName("java_4");classDo.setClassNo("java_4");classService.insertClass(classDo);}

执行结果:

解决方案:捕获异常并抛出异常

 @Override@Transactional(propagation = Propagation.REQUIRED,rollbackFor = Exception.class)public void insertClassByException(ClassDo classDo) {classMapper.insertClass(classDo);try {int i = 1 / 0;} catch (Exception e) {e.printStackTrace();throw new RuntimeException();}}

事务传播行为设置异常

此种事务传播行为不是特殊自定义设置,基本上不会使用Propagation.NOT_SUPPORTED,不支持事务

 @Transactional(propagation = Propagation.NOT_SUPPORTED,rollbackFor = Exception.class)public void insertClassByException(ClassDo classDo) {classMapper.insertClass(classDo);try {int i = 1 / 0;} catch (Exception e) {e.printStackTrace();throw new RuntimeException();}}

数据库存储引擎不支持事务

以MySQL关系型数据为例,如果其存储引擎设置为 MyISAM,则事务失效,因为MyISMA 引擎是不支持事务操作的;

故若要事务生效,则需要设置存储引擎为InnoDB ;目前 MySQL 从5.5.5版本开始默认存储引擎是:InnoDB;

推荐好文

>>【练手项目】基于SpringBoot的ERP系统,自带进销存+财务+生产功能>>分享一套基于SpringBoot和Vue的企业级中后台开源项目,代码很规范!
>>能挣钱的,开源 SpringBoot 商城系统,功能超全,超漂亮!

面试官:Spring事务失效的场景有哪些?如何解决?相关推荐

  1. 8个Spring事务失效的场景,你碰到过几种?

    前言 作为Java开发工程师,相信大家对Spring种事务的使用并不陌生.但是你可能只是停留在基础的使用层面上,在遇到一些比较特殊的场景,事务可能没有生效,直接在生产上暴露了,这可能就会导致比较严重的 ...

  2. Spring事务失效常见场景

    一.事务方法访问修饰符非public,导致事务失效 1.实例 2.解决 方式一:将方法修饰符改为public 方式二:开启AspectJ代理模式 3.注意 如果事务是static.final的,同样无 ...

  3. 详细整理Spring事务失效的具体场景及解决方案

    实际项目开发中,如果涉及到多张表操作时,为了保证业务数据的一致性,大家一般都会采用事务机制:好多小伙伴可能只是简单了解一下,遇到事务失效的情况,便会无从下手,溪源此篇文章给大家整理了一下常见Sprin ...

  4. 你否有遇到Spring事务失效,花费太多时间找bug

    作者:溪~源 来源:blog.csdn.net/xuan_lu/article/details/107797505 实际项目开发中,如果涉及到多张表操作时,为了保证业务数据的一致性,大家一般都会采用事 ...

  5. Spring事务失效的 8 大原因,这次可以吊打面试官了!

    今天再来一篇<吊打面试官>系列,这次真的要吊打了,哈哈!(看往期吊打系列请在后台回复:吊打,我会陆续更新--) 前几天栈长不是发了一篇文章,里面有一个关于事务失效的问题: 用 Spring ...

  6. 面试官:你知道哪几种事务失效的场景?

    前言 声明式事务是Spring功能中最爽之一,可是有些时候,我们在使用声明式事务并未生效,这是为什么呢? 今天陈某带大家来聊一聊声明事务的几种失效场景.本文将会从以下两个方面来说一下事务为什么会失效? ...

  7. Spring 事务失效的 8 大场景,面试官直呼666...

    前几天发了一篇文章里面有一个关于事务失效的问题: 用 Spring 的 @Transactional 注解控制事务有哪些不生效的场景? 其中有个热心粉丝留言分享了下,我觉得总结得有点经验,给置顶了: ...

  8. 详解spring事务失效和回滚失败的场景

    详解spring事务失效和回滚失败的场景 详解spring事务失效和回滚失败的场景 前言 一 .事务不生效 1.访问权限问题 2. 方法用final修饰 3.方法的内部调用 3.1 新加一个Servi ...

  9. Spring 事务失效?看这篇文章就够了!

    欢迎关注方志朋的博客,回复"666"获面试宝典 用 Spring 的 @Transactional 注解控制事务有哪些不生效的场景? 不知道小伙伴们有没有这样的经历,在自己开心的编 ...

最新文章

  1. 基于 Prometheus、InfluxDB 与 Grafana 打造监控平台
  2. 改变,从跨出第一步開始——记海大ITAEM团队首次IT讲座掠影
  3. 【pointnet++点云识别】基于pointnet++的点云地理数据识别的MATLAB仿真
  4. DM365 dvsdk_2_10_01_18开发环境搭建
  5. HDU1407 测试你是否和LTC水平一样高 暴力、二分、hash
  6. 发票抬头是什么意思?
  7. 使用Cloudformation集成Spring Boot和EC2
  8. 算法转换c语言程序,(转)C语言实现卡尔曼滤波算法程序
  9. UI-UIButton、UILable、UITextField总结
  10. linux 标准输出流管道,初学Linux之标准 I/O 和管道
  11. 移动端上下拖动调整顺序效果_HTML5 移动端的上下左右滑动问题
  12. 7200 笔记本硬盘 444
  13. 计算机桌面文件保存位置是哪里,电脑微信接收文件存放位置在哪?怎么更改文件存放位置...
  14. 【网络编程实践】2.4.2 muduo库安装与 procmon 编译
  15. H5网页元素和全局属性
  16. Java(JavaEE)学习线路图
  17. Javascript特效之可翻阅上一条下一条的动态文字
  18. Paper简读 - ProGen: Language Modeling for Protein Generation
  19. 形容计算机专业的诗句,形容人专业敬业的古诗句
  20. SNN系列|神经元模型篇(3)SRM

热门文章

  1. 我太难了!男子打赏女主播被骗9.6万,求助“好心人”再度被骗...
  2. 何小鹏“维权”事后谈造车:心很累 曾购上千瓶白酒缓解压力
  3. 苹果发布iOS 12.4首个测试版 苹果信用卡即将来袭
  4. 红米7正式首发:4GB+64GB版本售价999元
  5. 开挂了!女生用写字机器人刷作业 因写太快被识破 这能怪孩子吗?
  6. 什么是WEB?如何学习web
  7. Promise【面试】
  8. 晨哥真有料丨一定要在一个月内让她爱上你!
  9. python中argmin函数_Python numpy.argmin()用法及代码示例
  10. python字符串出栈方法_Python 实现字符串反转的9种方法