就这?Spring 事务失效场景及解决方案
小明:靓仔,我最近遇到了很邪门的事。
靓仔:哦?说来听听。
小明:上次看了你的文章《就这?一篇文章让你读懂 Spring 事务》,对事务有了详细的了解,但是在项目中还是遇到了问题,明明加了事务注解 @Transactional,却没有生效。
靓仔:那今天我就给你总结下哪些场景下事务会失效。
1、数据库引擎不支持事务
Mysql 常用的数据库引擎有 InnoDB 和 MyISAM,其中前者是支持事务的,而后者并不支持,MySQL 5.5.5 以前的默认存储引擎是:MyISAM,之前的版本默认的都是:InnoDB ,所以一定要注意自己使用的数据库支不支持事务。
2、没有被 Spring 管理
事务方法所在的类没有被注入Spring 容器,比如下面这样:
public class OrderServiceImpl implements OrderService {
@Autowired
AccountMapper accountMapper;
@Autowired
ProductMapper productMapper;
@Transactional
@Override
public void placeOrder() {
// 此处省略一堆逻辑
// 修用户改余额和商品库存
accountMapper.update();
productMapper.update();
}
}
这个类没有加 @service 注解,事务是不会生效的。
3、不是 public 方法
官方文档上已经说的很清楚了,@Transactional 注解只能用于 public 方法,如果要用在非 public 方法上,可以开启 AspectJ 代理模式。
4、异常被捕获
比如下面这个例子:
@Service
public class OrderServiceImpl implements OrderService {
@Autowired
AccountMapper accountMapper;
@Autowired
ProductMapper productMapper;
@Transactional
@Override
public void placeOrder() {
try{
// 此处省略一堆逻辑
// 修用户改余额和商品库存
accountMapper.update();
productMapper.update();
} catch (Exception e) {
}
}
}
当该方法发生异常的时候,由于异常被捕获,并没有抛出来,所以事务会失效,那这种情况下该怎么解决呢?别急,往下看
@Service
public class OrderServiceImpl implements OrderService {
@Autowired
AccountMapper accountMapper;
@Autowired
ProductMapper productMapper;
@Transactional
@Override
public void placeOrder() {
try{
// 此处省略一堆逻辑
// 修用户改余额和商品库存
accountMapper.update();
productMapper.update();
} catch (Exception e) {
// 手动回滚
TransactionAspectSupport.crrentTransactionStatus().setRollbackOnly();
}
}
}
可以通过
TransactionAspectSupport.crrentTransactionStatus().setRollbackOnly();
手动进行回滚操作。
5、异常类型错误
@Transactional 注解默认只回滚 RuntimeException 类型的异常,所以在使用的时候建议修改成 Exception 类型
@Transactional(rollbackFor = Exception.class)
6、内部调用事务方法
这应该是最常见的事务失效的的场景了吧,也是我要重点讲的情况。
有些业务逻辑比较复杂的操作,比如前面例子中的下单方法,往往在写操作之前会有一堆逻辑,如果所有操作都放在一个方法里,并且加上事务,那么很可能会因为事务执行时间过长,导致事务超时,就算没超时也会影响下单接口的性能。这时可以将写操作提取出来,只对写操作加上事务,那么压力就会小很多。
请看下面这个例子:
@Service
public class OrderServiceImpl implements OrderService {
@Autowired
AccountMapper accountMapper;
@Autowired
ProductMapper productMapper;
@Override
public void placeOrder() {
// 此处省略一堆逻辑
this.updateByTransactional();
}
@Transactional
public void updateByTransactional() {
// 修用户改余额和商品库存
accountMapper.update();
productMapper.update();
}
}
由于发生了内部调用,而没有经过 Spring 的代理,事务就不会生效,官方文档中也有说明:
那这种情况下该怎么办呢?
方案一:改为外部调用
内部调用不行,那我改成外部调用不就行了么
@Service
public class OrderServiceImpl implements OrderService {
@Autowired
OrderTransactionService orderTransactionService;
@Override
public void placeOrder() {
// 此处省略一堆逻辑
orderTransactionService.updateByTransactional();
}
}
@Service
public class OrderTransactionService {
@Autowired
AccountMapper accountMapper;
@Autowired
ProductMapper productMapper;
@Transactional
public void updateByTransactional() {
// 修用户改余额和商品库存
accountMapper.update();
productMapper.update();
}
}
这是比较容易理解的一种方法
方案二:使用编程式事务
既然声明式事务有问题,那我换成编程式事务可还行?
@Service
public class OrderServiceImpl implements OrderService {
@Autowired
AccountMapper accountMapper;
@Autowired
ProductMapper productMapper;
@Autowired
TransactionTemplate transactionTemplate;
@Override
public void placeOrder() {
// 此处省略一堆逻辑
// TransactionCallbackWithoutResult 无返回参数
// TransactionCallback 有返回参数
transactionTemplate.execute(new TransactionCallbackWithoutResult() {
@Override
protected void doInTransactionWithoutResult(TransactionStatus transactionStatus) {
try {
this.updateByTransactional();
} catch (Exception e) {
log.error("下单失败", e);
transactionStatus.setRollbackOnly();
}
}
});
}
public void updateByTransactional() {
// 修用户改余额和商品库存
accountMapper.update();
productMapper.update();
}
}
甭管他黑猫白猫,能抓住老鼠的就是好猫
方案三:通过外部方法调回来
这个是我看到网友提供的一种方法,又想用注解,又想自调用,那么可以参考编程式事务的方式来实现。
@Component
public class TransactionComponent {
public interface Callback<T>{
T run() throws Exception;
}
public interface CallbackWithOutResult {
void run() throws Exception;
}
// 带返回参数
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
@Nullable
public <T> T doTransactional(Callback<T> callback) throws Exception {
return callback.run();
}
// 无返回参数
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
@Nullable
public void doTransactionalWithOutResult(CallbackWithOutResult callbackWithOutResult) throws Exception {
callbackWithOutResult.run();
}
}
这样通过 TransactionComponent 调用内部方法,就可以解决失效问题了。
@Service
public class OrderServiceImpl implements OrderService {
@Autowired
AccountMapper accountMapper;
@Autowired
ProductMapper productMapper;
@Autowired
TransactionComponent transactionComponent;
@Override
public void placeOrder() {
// 此处省略一堆逻辑
transactionComponent.doTransactionalWithOutResult(() -> this.updateByTransactional());
}
public void updateByTransactional() {
// 修用户改余额和商品库存
accountMapper.update();
productMapper.update();
}
}
总结
本文总结了比较常见的几种事务失效的场景,以及一些解决方案,不一定很全。你还遇到了哪些我没提到的场景,欢迎分享,有不足之处,也欢迎指正。
END
往期推荐
就这?Spring 事务失效场景及解决方案的更多相关文章
- Spring事务失效的原因
http://blog.csdn.net/paincupid/article/details/51822599 Spring事务失效的原因 5种大的原因 如使用mysql且引擎是MyISAM,则事务会 ...
- Spring事务失效的2种情况
使用默认的事务处理方式 因为在java的设计中,它认为不继承RuntimeException的异常是”checkException”或普通异常,如IOException,这些异常在java语法中是要求 ...
- java面试记录二:spring加载流程、springmvc请求流程、spring事务失效、synchronized和volatile、JMM和JVM模型、二分查找的实现、垃圾收集器、控制台顺序打印ABC的三种线程实现
注:部分答案引用网络文章 简答题 1.Spring项目启动后的加载流程 (1)使用spring框架的web项目,在tomcat下,是根据web.xml来启动的.web.xml中负责配置启动spring ...
- 聊聊spring事务失效的12种场景,太坑了
前言 对于从事java开发工作的同学来说,spring的事务肯定再熟悉不过了. 在某些业务场景下,如果一个请求中,需要同时写入多张表的数据.为了保证操作的原子性(要么同时成功,要么同时失败),避免数据 ...
- spring事务失效的12种场景
一 事务不生效 1.访问权限问题 java的访问权限主要有四种:private<default<protected<public. 把有某些事务方法,定义了错误的访问权限,就会导致事 ...
- Spring事务失效的 8 大原因,这次可以吊打面试官了!
今天再来一篇<吊打面试官>系列,这次真的要吊打了,哈哈!(看往期吊打系列请在后台回复:吊打,我会陆续更新--) 前几天栈长不是发了一篇文章,里面有一个关于事务失效的问题: 用 Spring ...
- spring事务失效情况分析
详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt113 <!--[if !supportLists]-->一.&l ...
- Spring事务深入剖析--spring事务失效的原因
之前我们讲的分布式事务的调用都是在一个service中的事务方法,去调用另外一个service中的业务方法, 如果在一个sevice中存在两个分布式事务方法,在一个seivice中两个事务方法相互嵌套 ...
- Spring 事务失效
隔离级别 在 TransactionDefinition.java 接口中,定义了"四种"的隔离级别枚举: /** * [Spring 独有]使用后端数据库默认的隔离级别 * * ...
随机推荐
- 入“坑”mybatis后如何挣脱?
既然已经入"坑"mybatis了,你竟然还想着挣脱,我是不会让你挣脱的~ 当然我有一个算是挣脱的办法.那就是把它学会.理解透.这样我们也就不用在坑里一直徘徊,也算得上是一种挣脱吧! ...
- 手把手教你IDEA连接码云(Gitee)
目录 前言 一.下载.安装git 1.打开git官网,选择你的操作系统 2.根据你的系统位数选择相应的版本下载 3.安装 4.配置全局的用户名.邮箱 5.在idea中配置git目录 二.配置Gitee ...
- 关于WLS2中Ubuntu启用SSH远程登录功能,基于Xshell登录,支持Root
背景介绍 虽然WSL2提供了非常便利的访问Ubuntu目录的形式,但是仍然我们需要通过一个工具,比如XSHELL来实现对Ubuntu的SSH登录. 获取并安装Xshell 7 目前Xshell已经更新 ...
- AcWing 1143. 联络员
Tyvj已经一岁了,网站也由最初的几个用户增加到了上万个用户,随着Tyvj网站的逐步壮大,管理员的数目也越来越多,现在你身为Tyvj管理层的联络员,希望你找到一些通信渠道,使得管理员两两都可以联络(直 ...
- Linux使用shell脚本监控
(1)性能监控脚本 performance.sh #!/bin/bash #-------------------------------------------------------------- ...
- 记一次Struts中文乱码
起因 最近公司一个智能家具的项目,需要开发后端,APP/WEB的所有请求通过HTPP发送到后台,后台通过socket连接到智能设备.公司只有一个Java技术栈的同事,而他负责设备方面,我只能赶鸭子上架 ...
- SpringCloud的Ribbon自定义负载均衡算法
1.Ribbon默认使用RoundRobinRule策略轮询选择server 策略名 策略声明 策略描述 实现说明 BestAvailableRule public class BestAvailab ...
- vim下出现^M怎么解决
将window下的文本文件上传到linux上,在读取数据文件时,在每一行数据后会出现^M字符. 为什么会出现这种情况呢: 因为windows.linux.os系统的换行符标准不同: 先了解下概念, ...
- 关于中文版的manpages
可以从下面下载对应的包: https://code.google.com/p/manpages-zh/ https://github.com/lidaobing/manpages-zh 目前只有一部分 ...
- git使用---安装,提交,回退,修改,分支,标签等
下面是对git的各种使用及命令的基础使用,来自廖雪峰老师的git教程,这个收录下,作为git的使用总结. github上面地址为:https://github.com/Zhangguoliu/lear ...