点击上方蓝色“方志朋”,选择“设为星标”

回复“666”获取独家整理的学习资料!

事务管理在系统开发中是不可缺少的一部分,Spring提供了很好事务管理机制,主要分为编程式事务和声明式事务两种。

关于事务的基础知识,如什么是事务,数据库事务以及Spring事务的ACID、隔离级别、传播机制、行为等,就不在这篇文章中详细介绍了。默认大家都有一定的了解。

本文,作者会先简单介绍下什么是声明式事务和编程式事务,再说一下为什么我不建议使用声明式事务。

编程式事务

基于底层的API,如PlatformTransactionManager、TransactionDefinition 和 TransactionTemplate 等核心接口,开发者完全可以通过编程的方式来进行事务管理。

编程式事务方式需要是开发者在代码中手动的管理事务的开启、提交、回滚等操作。

public void test() {TransactionDefinition def = new DefaultTransactionDefinition();TransactionStatus status = transactionManager.getTransaction(def);try {// 事务操作// 事务提交transactionManager.commit(status);} catch (DataAccessException e) {// 事务提交transactionManager.rollback(status);throw e;}}

如以上代码,开发者可以通过API自己控制事务。

声明式事务

声明式事务管理方法允许开发者配置的帮助下来管理事务,而不需要依赖底层API进行硬编码。开发者可以只使用注解或基于配置的 XML 来管理事务。

@Transactionalpublic void test() {// 事务操作  }

如上,使用@Transactional即可给test方法增加事务控制。

当然,上面的代码只是简化后的,想要使用事务还需要一些配置内容。这里就不详细阐述了。

这两种事务,格子有各自的优缺点,那么,各自有哪些适合的场景呢?为什么有人会拒绝使用声明式事务呢?

声明式事务的优点

通过上面的例子,其实我们可以很容易的看出来,声明式事务帮助我们节省了很多代码,他会自动帮我们进行事务的开启、提交以及回滚等操作,把程序员从事务管理中解放出来。

声明式事务管理使用了 AOP 实现的,本质就是在目标方法执行前后进行拦截。 在目标方法执行前加入或创建一个事务,在执行方法执行后,根据实际情况选择提交或是回滚事务。

使用这种方式,对代码没有侵入性,方法内只需要写业务逻辑就可以了。

但是,声明式事务真的有这么好么?倒也不见得。

声明式事务的粒度问题

首先,声明式事务有一个局限,那就是他的最小粒度要作用在方法上。

也就是说,如果想要给一部分代码块增加事务的话,那就需要把这个部分代码块单独独立出来作为一个方法。

但是,正是因为这个粒度问题,本人并不建议过度的使用声明式事务。

首先,因为声明式事务是通过注解的,有些时候还可以通过配置实现,这就会导致一个问题,那就是这个事务有可能被开发者忽略。

事务被忽略了有什么问题呢?

首先,如果开发者没有注意到一个方法是被事务嵌套的,那么就可能会再方法中加入一些如RPC远程调用、消息发送、缓存更新、文件写入等操作。

我们知道,这些操作如果被包在事务中,有两个问题:

1、这些操作自身是无法回滚的,这就会导致数据的不一致。可能RPC调用成功了,但是本地事务回滚了,可是PRC调用无法回滚了。

2、在事务中有远程调用,就会拉长整个事务。那么久会导致本事务的数据库连接一直被占用,那么如果类似操作过多,就会导致数据库连接池耗尽。

有些时候,即使没有在事务中进行远程操作,但是有些人还是可能会不经意的进行一些内存操作,如运算。或者如果遇到分库分表的情况,有可能不经意间进行跨库操作。

但是如果是编程式事务的话,业务代码中就会清清楚楚看到什么地方开启事务,什么地方提交,什么时候回滚。这样有人改这段代码的时候,就会强制他考虑要加的代码是否应该方法事务内。

有些人可能会说,已经有了声明式事务,但是写代码的人没注意,这能怪谁。

话虽然是这么说,但是我们还是希望可以通过一些机制或者规范,降低这些问题发生的概率。

比如建议大家使用编程式事务,而不是声明式事务。因为,作者工作这么多年来,发生过不止一次开发者没注意到声明式事务而导致的故障。

因为有些时候,声明式事务确实不够明显。

声明式事务用不对容易失效

除了事务的粒度问题,还有一个问题那就是声明式事务虽然看上去帮我们简化了很多代码,但是一旦没用对,也很容易导致事务失效。

如以下几种场景就可能导致声明式事务失效:

1、@Transactional 应用在非 public 修饰的方法上

2、@Transactional 注解属性 propagation 设置错误

3、@Transactional 注解属性 rollbackFor 设置错误

4、同一个类中方法调用,导致@Transactional失效

5、异常被catch捕获导致@Transactional失效

6、数据库引擎不支持事务

以上几个问题,如果使用编程式事务的话,很多都是可以避免的。

使用声明事务失效的问题我们发生过很多次。不知道大家有没有遇到过,我是实际遇到过的

因为Spring的事务是基于AOP实现的,但是在代码中,有时候我们会有很多切面,不同的切面可能会来处理不同的事情,多个切面之间可能会有相互影响。

在之前的一个项目中,我就发现我们的Service层的事务全都失效了,一个SQL执行失败后并没有回滚,排查下来才发现,是因为一位同事新增了一个切面,这个切面里面做个异常的统一捕获,导致事务的切面没有捕获到异常,导致事务无法回滚。

这样的问题,发生过不止一次,而且不容易被发现。

很多人还是会说,说到底还是自己能力不行,对事务理解不透彻,用错了能怪谁。

但是我还是那句话,我们确实无法保证所有人的能力都很高,也无法要求所有开发者都能不出错。我们能做的就是,尽量可以通过机制或者规范,来避免或者降低这些问题发生的概率。

其实,如果大家有认真看过阿里巴巴出的那份Java开发手册的话,其实就能发现,其中的很多规约并不是完完全全容易被人理解,有些也比较生硬,但是其实,这些规范都是从无数个坑里爬出来的开发者们总结出来的。

关于@Transactional的用法,规约中也有提到过,只不过规约中的观点没有我这么鲜明:

总结

最后,相信本文的观点很多人都并不一定认同,很多人会说:Spring官方都推荐无侵入性的声明式事务,你有啥资格出来BB 。

说实话,刚工作的前几年,我也热衷于使用声明式事务,觉得很干净,也很"优雅"。觉得师兄们使用编程式事务多此一举,没有工匠精神。

但是慢慢的,线上发生过几次问题之后,我们复盘后发现,很多时候你自己写的代码很优雅,这完全没问题。

但是,优雅的同时也带来了一些副作用,师兄们又不能批评我,因为我的用法确实没错…

所以,有些事,还是要痛过之后才知道。

当然,本文并不要求大家一定要彻底不使用声明式事务,只是建议大家日后在使用事务的时候,能够考虑到本文中提到的观点,然后自行选择。

热门内容:
  • Java生鲜电商平台-监控模块的设计与架构

  • 通用的底层埋点都是怎么做的?

  • Java身份证号码识别系统

最近面试BAT,整理一份面试资料《Java面试BAT通关手册》,覆盖了Java核心技术、JVM、Java并发、SSM、微服务、数据库、数据结构等等。获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。
明天见(。・ω・。)ノ♡

Spring官方都推荐使用的@Transactional事务,为啥我不建议使用!相关推荐

  1. Spring官方都说废掉GuavaCache用Caffeine,你还不换?

    最近来了一个实习生小张,看了我在公司项目中使用的缓存框架Caffeine,三天两头跑来找我取经,说是要把Caffeine吃透,为此无奈的也只能一个个细心解答了. 后来这件事情被总监直到了,说是后面还有 ...

  2. Spring提取@Transactional事务注解的源码解析

    声明:本文是自己在学习spring注解事务处理源代码时所留下的笔记: 难免有错误,敬请读者谅解!!! 1.事务注解标签 <tx:annotation-driven /> 2.tx 命名空间 ...

  3. 【Spring源码】Spring Transactional事务:传播机制(Propagation) 介绍 和 源码剖析

    [Spring源码]Spring Transactional事务:传播机制(Propagation) 源码剖析 关键词 AMethod调用BMethod,转载BMethod的角度来考虑:站在被调用者的 ...

  4. @Transactional事务的使用和注意事项及其属性

    事务管理 提示 @Transactional注解只能应用到public可见度的方法上,可以被应用于接口定义和接口方法,方法会覆盖类上面声明的事务. 示例: 例如用户新增需要插入用户表.用户与岗位关联表 ...

  5. Spring Data JPA 从入门到精通~事务的处理及其讲解

    默认 @Transactional 注解式事务 (1)@EnableTransactionManagement 正常情况下,我们是需要在 ApplicationConfig 类加上 @EnableTr ...

  6. Spring官方文档翻译(1~7章)

    Spring官方文档:http://docs.spring.io/spring/docs/current/spring-framework-reference/htmlsingle/ 一.Spring ...

  7. Spring官方文档翻译(1~6章)

    Spring官方文档:http://docs.spring.io/spring/docs/current/spring-framework-reference/htmlsingle/ #一.Sprin ...

  8. Spring官方文档翻译

    Spring官方文档:http://docs.spring.io/spring/docs/current/spring-framework-reference/htmlsingle/ 声明:此文章大部 ...

  9. Java的@Transactional事务回滚

    @Transactional 基本原理概述 在应用系统调用声明@Transactional 的目标方法时,Spring Framework 默认使用 AOP 代理,在代码运行时生成一个代理对象,根据@ ...

最新文章

  1. 美国商务部工业和安全局(BIS)发布《关于拟制定脑机接口技术出口管制规则的通知》...
  2. Linux 文件与目录管理、ls、cd、pwd、mkdir、rmdir、cp、 rm
  3. 关于 OutOfMemoryError 的总结与解决方法
  4. gridview不换行,高亮显示
  5. 学习记录——背包问题基础公式解释回顾
  6. 使用python的笔记
  7. VBA学习_3:对象、集合及对象的属性和方法
  8. global.asax、global.asax.compiled、PrecompiledApp.config三者关系
  9. 黄杏元《地理信息系统概论》考研复习考点精讲(一)
  10. 玩家开发、出售《劲舞团》外挂获刑一年半
  11. 《少有人走的路》语录
  12. 计算机之间的通信原理
  13. 项目进度管理-活动排序工具与技术(前导图、箭线图)
  14. 计算机视觉论文文献综述怎么找,如何有效在知网寻找属于自己课题的文献综述...
  15. Android wifi 破解
  16. 微信小游戏真机调试卡在100%
  17. 最新OPPOR9S无人直播魅族版硬改相机刷机教程
  18. 资金内部收益率怎么计算
  19. 抚仙湖,一个亦梦亦幻的地方,这个五一节,我们骑友′的诗和远方
  20. 学习笔记:unity通过Mesh网格绘制图形:三角形正方体圆柱

热门文章

  1. Isight 命令行运行任务
  2. C/s模式B/S模式
  3. jQuery学习笔记(一)
  4. 【青少年编程】【四级】用逗号分隔列表
  5. 《C#精彩实例教程》小组阅读01 – MSDN是什么?
  6. 利用“队列”解决“窗口混乱”问题
  7. iOS15.4 来袭:新增“男妈妈”表情及口罩面容解锁、AirTags 反跟踪等新功能
  8. 超级详细的 Python 数据分析指南
  9. 关于GCN,我有三种写法
  10. 利用 AssemblyAI 在 PyTorch 中建立端到端的语音识别模型