我是 ABin-阿斌:写一生代码,创一世佳话,筑一览芳华。 如果小伙伴们觉得文章有点 feel ,那就点个赞再走哦。

  • 声明:原文地址:微信公众号:楼仔

下一篇:Transactional 的具体使用

文章目录

  • 一、项目准备
  • 二、事务不生效的几种 Case
    • Case 1: 类内部访问
    • Case 2: 私有方法
    • Case 3: 异常不匹配
    • Case 4: 多线程
      • 父线程抛出异常
      • 子线程抛出异常
  • 三、源码解读
    • 1、@Transactional 执行机制
    • 2、private 导致事务不生效原因
    • 3、异常不匹配原因

这篇文章,会先讲述 @Transactional 的 4 种不生效的 Case,然后再通过源码解读,分析 @Transactional 的执行原理,以及部分 Case 不生效的真正原因。

一、项目准备

下面是 DB 数据和 DB 操作接口:

uid uname usex
1 张三
2 陈恒
3 楼仔
// 提供的接口
public interface UserDao {// select * from user_test where uid = "#{uid}"public MyUser selectUserById(Integer uid);// update user_test set uname =#{uname},usex = #{usex} where uid = #{uid}public int updateUser(MyUser user);
}

基础测试代码,testSuccess() 是事务生效的情况:

@Service
public class UserController {@Autowiredprivate UserDao userDao;public void update(Integer id) {MyUser user = new MyUser();user.setUid(id);user.setUname("张三-testing");user.setUsex("女");userDao.updateUser(user);}public MyUser query(Integer id) {MyUser user = userDao.selectUserById(id);return user;}// 正常情况@Transactional(rollbackFor = Exception.class)public void testSuccess() throws Exception {Integer id = 1;MyUser user = query(id);System.out.println("原记录:" + user);update(id);throw new Exception("事务生效");}
}

二、事务不生效的几种 Case

主要讲解 4 种事务不生效的 Case:

  • 类内部访问:A 类的 a1 方法没有标注 @Transactional,a2 方法标注 @Transactional,在 a1 里面调用 a2;

  • 私有方法:将 @Transactional 注解标注在非 public 方法上;

  • 异常不匹配:@Transactional 未设置 rollbackFor 属性,方法返回 Exception 等异常;

  • 多线程:主线程和子线程的调用,线程抛出异常。

Case 1: 类内部访问

我们在类 UserController 中新增一个方法 testInteralCall():

public void testInteralCall() throws Exception {testSuccess();throw new Exception("事务不生效:类内部访问");
}

这里 testInteralCall() 没有标注 @Transactional,我们再看一下测试用例:

public static void main(String[] args) throws Exception {ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");UserController uc = (UserController) applicationContext.getBean("userController");try {uc.testSuccess();} finally {MyUser user =  uc.query(1);System.out.println("修改后的记录:" + user);}
}
// 输出:
// 原记录:MyUser(uid=1, uname=张三, usex=女)
// 修改后的记录:MyUser(uid=1, uname=张三-testing, usex=女)

从上面的输出可以看到,事务并没有回滚,这个是什么原因呢?

因为 @Transactional 的工作机制是基于 AOP 实现,AOP 是使用动态代理实现的,如果通过代理直接调用 testSuccess(),通过 AOP 会前后进行增强,增强的逻辑其实就是在 testSuccess() 的前后分别加上开启、提交事务的逻辑,后面的源码会进行剖析。

现在是通过 testInteralCall() 去调用 testSuccess(),testSuccess() 前后不会进行任何增强操作,也就是类内部调用,不会通过代理方式访问。

如果还是不太清楚,推荐再看看这篇文章,里面有完整示例,非常完美诠释“类内部访问”不能前后增强的原因:https://blog.csdn.net/Ahuuua/article/details/123877835

Case 2: 私有方法

在私有方法上,添加 @Transactional 注解也不会生效:

@Transactional(rollbackFor = Exception.class)
private void testPirvateMethod() throws Exception {Integer id = 1;MyUser user = query(id);System.out.println("原记录:" + user);update(id);throw new Exception("测试事务生效");
}

直接使用时,下面这种场景不太容易出现,因为 IDEA 会有提醒,文案为: Methods annotated with ‘@Transactional’ must be overridable,至于深层次的原理,源码部分会给你解读。

Case 3: 异常不匹配

这里的 @Transactional 没有设置 rollbackFor = Exception.class 属性:

@Transactional
public void testExceptionNotMatch() throws Exception {Integer id = 1;MyUser user = query(id);System.out.println("原记录:" + user);update(id);throw new Exception("事务不生效:异常不匹配");
}

测试方法:同 Case1

  • // 输出:
  • // 原记录:User[uid=1,uname=张三,usex=女]
  • // 修改后的记录:User[uid=1,uname=张三-test,usex=女]

@Transactional 注解默认处理运行时异常,即只有抛出运行时异常时,才会触发事务回滚,否则并不会回滚,至于深层次的原理,源码部分会给你解读。

Case 4: 多线程

下面给出两个不同的姿势,一个是子线程抛异常,主线程 ok;一个是子线程 ok,主线程抛异常。

父线程抛出异常

父线程抛出异常,子线程不抛出异常:

public void testSuccess() throws Exception {Integer id = 1;MyUser user = query(id);System.out.println("原记录:" + user);update(id);
}
@Transactional(rollbackFor = Exception.class)
public void testMultThread() throws Exception {new Thread(new Runnable() {@SneakyThrows@Overridepublic void run() {testSuccess();}}).start();throw new Exception("测试事务不生效");
}

父线程抛出线程,事务回滚,因为子线程是独立存在,和父线程不在同一个事务中,所以子线程的修改并不会被回滚,

子线程抛出异常

父线程不抛出异常,子线程抛出异常:

public void testSuccess() throws Exception {Integer id = 1;MyUser user = query(id);System.out.println("原记录:" + user);update(id);throw new Exception("测试事务不生效");
}
@Transactional(rollbackFor = Exception.class)
public void testMultThread() throws Exception {new Thread(new Runnable() {@SneakyThrows@Overridepublic void run() {testSuccess();}}).start();
}

由于子线程的异常不会被外部的线程捕获,所以父线程不抛异常,事务回滚没有生效。

三、源码解读

下面我们从源码的角度,对 @Transactional 的执行机制和事务不生效的原因进行解读。

1、@Transactional 执行机制

我们只看最核心的逻辑,代码中的 interceptorOrInterceptionAdvice 就是 TransactionInterceptor 的实例,入参是 this 对象。

红色方框有一段注释,大致翻译为 “它是一个拦截器,所以我们只需调用即可:在构造此对象之前,将静态地计算切入点。”

this 是 ReflectiveMethodInvocation 对象,成员对象包含 UserController 类、testSuccess() 方法、入参和代理对象等。

进入 invoke() 方法后:

前方高能!!!这里就是事务的核心逻辑,包括判断事务是否开启、目标方法执行、事务回滚、事务提交。

2、private 导致事务不生效原因

在上面这幅图中,第一个红框区域调用了方法 getTransactionAttribute(),主要是为了获取 txAttr 变量,它是用于读取 @Transactional 的配置,如果这个 txAttr = null,后面就不会走事务逻辑,我们看一下这个变量的含义:

我们直接进入 getTransactionAttribute(),重点关注获取事务配置的方法。

前方高能!!!这里就是 private 导致事务不生效的原因所在,allowPublicMethodsOnly() 一直返回 false,所以重点只关注 isPublic() 方法。

下面通过位与计算,判断是否为 Public,对应的几类修饰符如下:

  • PUBLIC: 1

  • PRIVATE: 2

  • PROTECTED: 4

看到这里,是不是豁然开朗了,有没有觉得很有意思呢~~

3、异常不匹配原因

我们继续回到事务的核心逻辑,因为主方法抛出 Exception() 异常,进入事务回滚的逻辑:

进入 rollbackOn() 方法,判断该异常是否能进行回滚,这个需要判断主方法抛出的 Exception() 异常,是否在 @Transactional 的配置中:

我们进入 getDepth() 看一下异常规则匹配逻辑,因为我们对 @Transactional 配置了 rollbackFor = Exception.class,所以能匹配成功:

示例中的 winner 不为 null,所以会跳过下面的环节。但是当 winner = null 时,也就是没有设置 rollbackFor 属性时,会走默认的异常捕获方式。

前方高能!!!这里就是异常不匹配原因的原因所在,我们看一下默认的异常捕获方式:

是不是豁然开朗,当没有设置 rollbackFor 属性时,默认只对 RuntimeException 和 Error 的异常执行回滚。

Spring事务:一文带你升入 @Transactional 底层相关推荐

  1. 【Spring】一文带你吃透基于XML的DI技术

    个人主页: 几分醉意的CSDN博客_传送门 文章目录

  2. 【Spring】一文带你吃透基于注解的DI技术

    个人主页: 几分醉意的CSDN博客_传送门 本文目录

  3. Spring事务源码分析

    首先看例子,这例子摘抄自开涛的跟我学spring3. @Test public void testPlatformTransactionManager() { DefaultTransactionDe ...

  4. Spring事务报错: org.springframework.transaction.UnexpectedRollbackException

    异常信息: 出现了不可预知的回滚异常,因为事务已经被标志位只能回滚,所以事务回滚了. org.springframework.transaction.UnexpectedRollbackExcepti ...

  5. Spring事务隔离级别与数据库隔离级别不一致时,该以谁为准?

    原创博文,欢迎转载,转载时请务必附上博文链接,感谢您的尊重. 前言 通过本篇,你将了解到[Spring事务]与[数据库事务]的关系,以及优先级问题,我将为你一一论证. 阅读本篇,你可能会需要的博文: ...

  6. Spring事务五边形

    ACID:原子性(Atomicity).一致性(Consistency).隔离性(Isolation).持久性(Durability) 事务五边形 传播行为 隔离级别 是否只读 事务超时时间 回滚规则 ...

  7. 一文带你看懂Spring事务!

    点击上方"方志朋",选择"设为星标" 做积极的人,而不是积极废人 前言 Spring事务管理我相信大家都用得很多,但可能仅仅局限于一个@Transactiona ...

  8. 一文带你认识Spring事务

    前言 只有光头才能变强. 文本已收录至我的GitHub仓库,欢迎Star:https://github.com/ZhongFuCheng3y/3y Spring事务管理我相信大家都用得很多,但可能仅仅 ...

  9. 一文带你深入理解 Spring 事务原理

    点击上方 "程序员小乐"关注, 星标或置顶一起成长 后台回复"大礼包"有惊喜礼包! 关注订阅号「程序员小乐」,收看更多精彩内容 每日英文 Man has to ...

最新文章

  1. 格雷码编码+解码+实现(Python)
  2. vs2008格式化代码
  3. Spring Boot + BeetlSQL + H2数据库项目整合
  4. [导入]将DataGrid输出到Excel文件
  5. 004_Mysql数据库的CRUD的操作
  6. 即时通讯音视频开发(七):音频基础及编码原理入门
  7. 整合spring cloud云架构 - SSO单点登录之OAuth2.0 登出流程(3)
  8. JavaScript复习使用定时器的简易式诸葛大力轮播图
  9. python如何运行源文件_Python如何运行
  10. django-后台管理-笔记
  11. C++二分查找示例(求货物装载量)
  12. 【转】计算机科学中最重要的32个算法
  13. 毫秒和秒的换算工具_使用后戒不掉的文档搜索工具:Everything
  14. java ognl表达式_常用的OGNL表达式
  15. vue3.0+vite跑项目遇到的问题
  16. Type-C边充电边OTG芯片LDR6028A
  17. java中国象棋兵吃棋规则_中国象棋吃子的规则
  18. 【报告分享】2021年中国互联网保险消费者洞察报告-凯度元保清华大学国家金融研究院(附下载)
  19. FPGA图像处理HLS实现sobel边沿检测,提供HLS工程和vivado工程源码
  20. 来看看科技公司们在今年的愚人节都开了什么玩笑

热门文章

  1. 树莓派,tx2硬件对比
  2. mx2 android os耗电,魅族MX2续航能力怎么样
  3. GMIC 2018大会AI 生万物 嘉宾分享摘要
  4. 树莓派-5-GPIO的应用实验
  5. stm32f103vet6通过L298N驱动12V直流无刷电机过程含代码
  6. 大学宿舍管理系统 C语言实现
  7. 基于JAVA至臻阁古董拍卖网计算机毕业设计源码+系统+mysql数据库+lw文档+部署
  8. 【操作系统⑫】——存储管理(下)【分段存储管理 虚拟存储管理 段页式存储管理方案 页面置换算法 OPT FIFO LRU】
  9. 基于Web技术的优秀电影片段赏析与交流系统
  10. mysql安装服务:Internal error (值不能为null.参数名:path1)The installer will now close