Java充电社
专辑
博文
联系我
本人继续续收门徒,亲手指导
Spring系列第51篇:导致 Spring 事务失效常见的几种情况
相关专辑:
Spring教程
|
Spring事务专题
<div style="display:none"></div> 这算是spring事务第9篇文章了,花了这么多篇文章介绍事务这块的知识,说明事务这块的东西确实比较多、知识点比较细,也非常重要,希望大家能够重视起来,吃透这块的知识。 本文2个目的: **1、使用spring事务的过程中,哪些情况会导致事务失效?** **2、遇到事务相关bug时,有哪些方法可以快速定位bug?** **3、文末有福利** ## 1、事务失效的7种情况 1. 未启用spring事务管理功能 2. 方法不是public类型的 3. 数据源未配置事务管理器 4. 自身调用问题 5. 异常类型错误 6. 异常被吞了 7. 业务和spring事务代码必须在一个线程中 ### 1.1、未启用spring事务管理功能 @EnableTransactionManagement 注解用来启用spring事务自动管理事务的功能,这个注解千万不要忘记写了。 ### 1.2、方法不是public类型的 @Transaction 可以用在类上、接口上、public方法上,如果将@Trasaction用在了非public方法上,事务将无效。 ### 1.3、数据源未配置事务管理器 spring是通过事务管理器了来管理事务的,一定不要忘记配置事务管理器了,要注意为每个数据源配置一个事务管理器: ```java @Bean public PlatformTransactionManager transactionManager(DataSource dataSource) { return new DataSourceTransactionManager(dataSource); } ``` ### 1.4、自身调用问题 spring是通过aop的方式,对需要spring管理事务的bean生成了代理对象,然后通过代理对象拦截了目标方法的执行,在方法前后添加了事务的功能,所以必须通过代理对象调用目标方法的时候,事务才会起效。 看下面代码,大家思考一个问题:当外部直接调用m1的时候,m2方法的事务会生效么? ```java @Component public class UserService { public void m1(){ this.m2(); } @Transactional public void m2(){ //执行db操作 } } ``` 显然不会生效,因为m1中通过this的方式调用了m2方法,而this并不是代理对象,this.m2()不会被事务拦截器,所以事务是无效的,如果外部直接调用通过UserService这个bean来调用m2方法,事务是有效的,上面代码可以做一下调整,如下,@1在UserService中注入了自己,此时m1中的m2事务是生效的 ```java @Component public class UserService { @Autowired //@1 private UserService userService; public void m1() { this.userService.m2(); } @Transactional public void m2() { //执行db操作 } } ``` **重点:必须通过代理对象访问方法,事务才会生效。** ### 1.5、异常类型错误 spring事务回滚的机制:对业务方法进行try catch,当捕获到有指定的异常时,spring自动对事务进行回滚,那么问题来了,哪些异常spring会回滚事务呢? 并不是任何异常情况下,spring都会回滚事务,默认情况下,RuntimeException和Error的情况下,spring事务才会回滚。 也可以自定义回滚的异常类型: ```java @Transactional(rollbackFor = {异常类型列表}) ``` ### 1.6、异常被吞了 当业务方法抛出异常,spring感知到异常的时候,才会做事务回滚的操作,若方法内部将异常给吞了,那么事务无法感知到异常了,事务就不会回滚了。 如下代码,事务操作2发生了异常,但是被捕获了,此时事务并不会被回滚 ```java @Transactional public void m1(){ 事务操作1 try{ 事务操作2,内部抛出了异常 }catch(Exception e){ } } ``` ### 1.7、业务和spring事务代码必须在一个线程中 spring事务实现中使用了ThreadLocal,ThreadLocal大家应该知道吧,可以实现同一个线程中数据共享,必须是同一个线程的时候,数据才可以共享,这就要求业务代码必须和spring事务的源码执行过程必须在一个线程中,才会受spring事务的控制,比如下面代码,方法内部的子线程内部执行的事务操作将不受m1方法上spring事务的控制,这个大家一定要注意 ```java @Transactional public void m1() { new Thread() { 一系列事务操作 }.start(); } ``` ## 2、如何快速定位事务相关bug? 2种方式 ### 方式1:看日志 如果你使用了logback或者log4j来输出日志,可以修改一下日志级别为debug模式,可以看到事务的详细执行日志,帮助你定位错误 ### 方式2:调试代码 如果你对源码比较了解,那么你会知道被spring管理事务的业务方法,执行的时候都会被TransactionInterceptor拦截器拦截,会进入到它的invoke方法中,咱们可以在invoke方法中设置一些断点,可以看到详细的执行过程,排错也就比较容易了。 整体上来说,还是需要你深入理解原理,原理了解了,写代码的时候本身就会避免很多坑。 <a style="display:none" target="_blank" href="https://mp.weixin.qq.com/s/_S1DD2JADnXvpexxaBwLLg" style="color:red; font-size:20px; font-weight:bold">继续收门徒,亲手带,月薪 4W 以下的可以来找我</a> ## 最新资料 1. <a href="https://mp.weixin.qq.com/s?__biz=MzkzOTI3Nzc0Mg==&mid=2247484964&idx=2&sn=c81bce2f26015ee0f9632ddc6c67df03&scene=21#wechat_redirect" target="_blank">尚硅谷 Java 学科全套教程(总 207.77GB)</a> 2. <a href="https://mp.weixin.qq.com/s?__biz=MzkwOTAyMTY2NA==&mid=2247484192&idx=1&sn=505f2faaa4cc911f553850667749bcbb&scene=21#wechat_redirect" target="_blank">2021 最新版 Java 微服务学习线路图 + 视频</a> 3. <a href="https://mp.weixin.qq.com/s?__biz=MzkwOTAyMTY2NA==&mid=2247484573&idx=1&sn=7f3d83892186c16c57bc0b99f03f1ffd&scene=21#wechat_redirect" target="_blank">阿里技术大佬整理的《Spring 学习笔记.pdf》</a> 4. <a href="https://mp.weixin.qq.com/s?__biz=MzkwOTAyMTY2NA==&mid=2247484544&idx=2&sn=c1dfe907cfaa5b9ae8e66fc247ccbe84&scene=21#wechat_redirect" target="_blank">阿里大佬的《MySQL 学习笔记高清.pdf》</a> 5. <a href="https://mp.weixin.qq.com/s?__biz=MzkwOTAyMTY2NA==&mid=2247485167&idx=1&sn=48d75c8e93e748235a3547f34921dfb7&scene=21#wechat_redirect" target="_blank">2021 版 java 高并发常见面试题汇总.pdf</a> 6. <a href="https://mp.weixin.qq.com/s?__biz=MzkwOTAyMTY2NA==&mid=2247485664&idx=1&sn=435f9f515a8f881642820d7790ad20ce&scene=21#wechat_redirect" target="_blank">Idea 快捷键大全.pdf</a> ![](https://itsoku.oss-cn-hangzhou.aliyuncs.com/itsoku/blog/article/1/2883e86e-3eff-404a-8943-0066e5e2b454.png)
相关专辑:
Spring教程
Spring事务专题