面试官问:导致 Spring 事务失效的场景有哪些,如何解决失效问题?

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip1024b (备注Java)
img

正文

public class StudentServiceImpl implements StudentService {

@Autowired

private StudentMapper studentMapper;

@Autowired

private 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直接调用,所以注解会失效。

@Service

public class ClassServiceImpl implements ClassService {

@Autowired

private 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();

}

}

//测试用例:

@Test

public void insertInnerExceptionTest() throws CustomException {

classDo.setClassId(2);

classDo.setClassName(“java_2”);

classDo.setClassNo(“java_2”);

classService.insertClass(classDo);

}

测试结果:

java.lang.RuntimeException

at 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 F a s t C l a s s B y S p r i n g C G L I B FastClassBySpringCGLIB FastClassBySpringCGLIBa1c03d8.invoke()

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

解决方案

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

public void insertClass(ClassDo classDo) throws CustomException {

//         insertClassByException(classDo);

((ClassServiceImpl)AopContext.currentProxy()).insertClassByException(classDo);

}

//测试用例:

@Test

public 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

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

@Service

public class ClassServiceImpl implements ClassService {

@Autowired

private 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();

}

//测试用例:

@Test

public void insertInnerExceptionTest() throws Exception {

classDo.setClassId(3);

classDo.setClassName(“java_3”);

classDo.setClassNo(“java_3”);

classService.insertClass(classDo);

}

}

运行结果:

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

java.lang.Exception

at com.qxy.common.service.impl.ClassServiceImpl.insertClassByException(ClassServiceImpl.java:35)

at com.qxy.common.service.impl.ClassServiceImpl F a s t C l a s s B y S p r i n g C G L I B FastClassBySpringCGLIB FastClassBySpringCGLIBa1c03d8.invoke()

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 {

@Autowired

private ClassMapper classMapper;

//    @Override

public 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();

}

}

}

最后:学习总结——MyBtis知识脑图(纯手绘xmind文档)

学完之后,若是想验收效果如何,其实最好的方法就是可自己去总结一下。比如我就会在学习完一个东西之后自己去手绘一份xmind文件的知识梳理大纲脑图,这样也可方便后续的复习,且都是自己的理解,相信随便瞟几眼就能迅速过完整个知识,脑补回来。下方即为我手绘的MyBtis知识脑图,由于是xmind文件,不好上传,所以小编将其以图片形式导出来传在此处,细节方面不是特别清晰。但可给感兴趣的朋友提供完整的MyBtis知识脑图原件(包括上方的面试解析xmind文档)

image

除此之外,前文所提及的Alibaba珍藏版mybatis手写文档以及一本小小的MyBatis源码分析文档——《MyBatis源码分析》等等相关的学习笔记文档,也皆可分享给认可的朋友!

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Java)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
晰。但可给感兴趣的朋友提供完整的MyBtis知识脑图原件(包括上方的面试解析xmind文档)

[外链图片转存中…(img-VSLNkFYr-1713559261752)]

除此之外,前文所提及的Alibaba珍藏版mybatis手写文档以及一本小小的MyBatis源码分析文档——《MyBatis源码分析》等等相关的学习笔记文档,也皆可分享给认可的朋友!

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Java)
[外链图片转存中…(img-0y5tfKws-1713559261753)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 10
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值