前言
相信大家都遇到一种事务失效场景,那就是 Spring 自调用,就是在 Service 方法内,调用另一个加 @Transactional 注解的方法,发现事务失效,这时候你是怎么解决的呢?
公众号:『 刘志航 』,记录工作学习中的技术、开发及源码笔记;时不时分享一些生活中的见闻感悟。欢迎大佬来指导!

事情回顾

那是一个我忘了天气咋样的下午,突然蹦出一个小红点,嗯~ 挺着急的小红点。

原来是事务失效了!

莫慌!莫慌!

最后小伙伴选择了抽走,是我的工具类不香了么?

当然故事的结果是完美的,问题解决了。

事务

在开发中涉及到同时操作多个表的时候,要保证两个操作要么一起成功,要么一起失败,这时候就需要用到事务。

现在一般使用的都是基于 @Transactional 注解的声明式事务

而事务使用过程中有以下几个注意事项:

  1. 事务只能应用到 public 方法上才会有效;
  2. 事务需要从外部调用,Spring 自调用会失效;
  3. 建议事务注解 @Transactional 一般添加在实现类上。

当然这几句话不是说我的,人家官方文档可是明确说明的!

这里可是说明了应仅将 @Transactional 注解应用于具有公开可见性的方法。如果对受 protected, private o或 package-visible 修饰的方法使用,则不会引发任何错误,但是被注解的方法不会显示已配置的事务设置。

说白了,就是你用了,不会报错,但是不生效!

至于建议加在实现类上,这个只是建议,不过如果加在接口类或接口方法上时,只有配置基于接口的代理才会生效。所以这块还是老老实实的加在实现类或实现类方法上吧。

因为代理模式只拦截通过代理传入的外部方法调用,所以自调用事务是不生效的。

官方的解释还是比较简单明了的,虽然我看不懂,但是不影响我截图。

那我还是再截一个吧……

实际使用

但是在开发中,小伙伴们往往会遇到这种情况!

本来自己写的代码就一坨坨的又臭又长,里面有各种验签、验参、查询、验证等等,就想着来个事务,让事务包裹的范围最小,仅仅在同时更新的时候加上事务吧!

这么写,咦~ IDEA 报错了,好像不能 private 修饰,那我改成 public

很显然事务是不生效的。

把更新的代码放到又臭又长的代码里面,让它变得更臭更长,然后用 @Transactional 注解一加。完美解决!

请放过那坨代码吧!来看看下面的办法。

解决方案 1

那我改成外部调用不就行了么?

再声明一个 Service,把更新表的逻辑放过去。

我一般就喜欢使用这个办法。

解决方案 2

使用编程式事务,前面说了,使用声明式事务时,又这又那,我换一种总可以吧!

你看,我还把方法改成 private 修饰了,事务也生效。完美解决!

其实这个方法也很不错哦!

解决方案 3

又想用注解,又想自调用怎么办?

不过... 麻烦一点还是可以的。

咱们可以参考编程式事务的方式,不就是不让自调用么,我调外部方法,然后外部方法再给我调回来不就可以了。

@Component
public class TransactionalComponent {public interface Cell {void run() throws Exception;}@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)public void required(Cell cell) throws Exception {cell.run();}
}

这样的话不就可以通过 TransactionalComponent 调用了么,并且还可以使用 lambda 表达式。

当然基于这个版本也可以做一个迭代,就是使用静态方法调用,不用每次都用 @Autowired 注入一次。

public class TransactionalUtils {private static volatile TransactionalComponent transactionalComponent;private static synchronized TransactionalComponent getTransactionalComponent() {if (transactionalComponent == null) {// 从容器中获取 transactionalComponenttransactionalComponent = ApplicationContextUtils.getBean(TransactionalComponent.class);}return transactionalComponent;}public static void required(TransactionalComponent.Cell cell) throws Exception {getTransactionalComponent().required(cell);}}

这样通过工具类 TransactionalUtils 便可以直接调用静态方法的方式执行事务操作。

总结

结束语

本文主要介绍为什么会遇到事务失效,以及事务失效的避免方式,同时提供了三种方式来解决自调用事务失效的问题。不足之处,欢迎指正。

相关资料

  1. Spring 文档:https://docs.spring.io/spring-framework/docs/5.3.0/reference/html/data-access.html#transaction-declarative-annotations

wordpress外部调用到html_Spring 自调用事务失效,你是怎么解决的?相关推荐

  1. html调用接口_Spring 自调用事务失效,你是怎么解决的?

    前言 " 相信大家都遇到一种事务失效场景,那就是 Spring 自调用,就是在 Service 方法内,调用另一个加 @Transactional 注解的方法,发现事务失效,这时候你是怎么解 ...

  2. Spring3.x事务失效的原因以及解决办法

    2019独角兽企业重金招聘Python工程师标准>>> 项目中如果使用spring来管理事务,可能会出现事务失效的情况,我认为主要的原因是cglib无法获取到代代理的实例.. 如果带 ...

  3. 事务失效了?别怕,这里有四种方式可以让他生效起来

    目录 一.背景 二.事务失效的原因 三.事务失效的真正原因 四.事务失效场景复现 五.事务失效的四种解决方案 六.留个思考题,和朋友探讨时留下的 一.背景 操作一张表或者多张表时,多次进行更新操作,确 ...

  4. springMVC重复扫描bean导致声明式事务失效

    文章目录 1 配置文件 1.1 加载spring容器配置 1.2 加载springMvc容器配置 1.3 spring声明式事务配置 2 声明式事务失效 2.1 事务失效的原因 2.2 解决方案 3 ...

  5. SpringBoot异常处理回滚事务详解(自动回滚、手动回滚、部分回滚)(事务失效)...

    参考:https://blog.csdn.net/zzhongcy/article/details/102893309 概念 事务定义 事务,就是一组操作数据库的动作集合.事务是现代数据库理论中的核心 ...

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

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

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

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

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

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

  9. spring中事务失效的几种情况

    下面简单介绍下,spring中常见的事务失效的几种情况.让我们在开发的过程避免这些情况,写出正确而且优雅的代码. 文章目录 数据库引擎不支持,mysql需要InnoDB 方法必须是public的 方法 ...

最新文章

  1. ORACLE等待事件:direct path write
  2. 大规模分布式消息中间件考虑点
  3. LC_ALL=C的含义
  4. qt geomery的单位是什么_斜管沉淀池的原理是什么?
  5. 《Android游戏开发详解》一导读
  6. python打开csv文件乱码_python脚本解决csv文件用excel打开乱码
  7. 单因素方差分析_基于R语言开展方差分析(一)——单因素方差分析
  8. php request对象,PHP 中TP5 Request 请求对象的实例详解
  9. bambook引起的adb启动异常的问题
  10. 20190813:(leetcode习题)加一
  11. 博客园使用攻略之如何添加自己的js文件
  12. MySQL(MariaDB)的 SSL 加密复制
  13. MOTO ME525/Defy 刷Android4.0 刷机教程
  14. ansible常用模块
  15. 饥荒联机版连不上服务器_《饥荒》无法连接klei服务器 刷不出服务器解决办法...
  16. 使用 KubeSphere 和极狐GitLab 打造云原生持续交付系统
  17. linux安装svn使用解压包的方式
  18. 微信公众号支付回调函数“time_end”的坑
  19. 计算机基础的建议,计算机应用基础教学实施建议.docx
  20. Hue安装、或操作过程中遇到的问题

热门文章

  1. 产品经理必备知识之网页设计系列(一)-创建出色用户体验
  2. 使用Selenium模拟浏览器,实现自动爬取数据
  3. ORACLE 新建数据库及权限赋予
  4. vector can通信源码_汽车电子CAN网络dbc文件
  5. matlab 小波变换_matlab小波工具箱实例(二):时频分析和连续小波变换
  6. react 调用组件方法_React源码分析1 — 组件和对象的创建(createClass,createElement)...
  7. LeetCode-剑指 Offer 12. 矩阵中的路径
  8. LeetCode-动态规划背包题-416. 分割等和子集
  9. QT一次性连接多个按钮槽函数
  10. QT中使用QSettings保存应用程序配置信息