前段时间写了关于CompletableFuture的使用博客,CompletableFuture使用方法详细说明CompletableFuture的thenCompose使用具体说明

但在实际中使用的时候发现,CompletableFuture开启的线程和当前事务是脱离开的,也就是当前上下文的事务和CompletableFuture中执行的事务是两个事务,谁也不影响谁,这样就会出现业务上的错乱。

例如:当前上下文执行失败,但是CompletableFuture中的执行成功了,而上下文因为失败进行了事务回滚,此时CompletableFuture中的是不会回滚的,这并不是我们想要的,所以针对这种情况进行测试说明和如何避免。

1.准备

1.1.表结构

CREATE TABLE `user` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(50) DEFAULT NULL COMMENT '名称',
`age` int(11) DEFAULT NULL COMMENT '年龄',
`address` varchar(50) DEFAULT NULL COMMENT '地址',
`tranmemo` varchar(100) DEFAULT NULL COMMENT '备注',
`createtime` datetime DEFAULT NULL COMMENT '创建时间',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8

插入几条测试数据:

insert into user values (null, 'zhangsan', 18, 'CHINA', '我是中国的张三', SYSDATE())
, (null, 'lisi', 28, 'CHINA', '我是中国的李四', SYSDATE())
, (null, 'wangwu', 38, 'CHINA', '我是中国的王五', SYSDATE())
, (null, 'tom', 5, 'USA', 'Tom and Jerry', SYSDATE())
, (null, 'jerry', 5, 'USA', 'Tom and Jerry', SYSDATE());

查询user

1.2.线程池

线程池参考 ThreadPoolTaskExecutor线程池创建

2.复现事务问题

2.1.模拟业务正常情况

创建CompletableFutureTransactionalController, 代码如下:

/**
* 测试CompletableFuture事务问题
*
* @author jiangkd
* @date 2022/10/07 15:44:31
*/
@RequiredArgsConstructor
@RequestMapping("/cf")
@RestController
public class CompletableFutureTransactionalController { private final CompletableFutureTransactionalService CompletableFutureTransactionalService; /**
* 根据id修改User, 测试CompletableFuture事务问题
*
* @param userId User主键值
*/
@PostMapping("/test/transactional")
public String updateUser(@RequestParam("userId") Integer userId) throws ExecutionException, InterruptedException {
//
CompletableFutureTransactionalService.updateUser(userId);
return String.valueOf(userId);
} }

创建CompletableFutureTransactionalService, 代码如下:

/**
* @author jiangkd
* @date 2022/10/07 16:08:44
*/
@RequiredArgsConstructor
@Slf4j
@Service
public class CompletableFutureTransactionalService { private final ThreadPoolTaskExecutor threadPoolTaskExecutor; private final UserDAO userDAO; /**
* 根据id修改User, 测试CompletableFuture事务问题
*
* @param userId User主键值
*/
@Transactional(rollbackFor = Exception.class)
public void updateUser(final Integer userId) throws ExecutionException, InterruptedException {
/*
创建CompletableFuture线程
*/
final CompletableFuture<Void> completableFuture = CompletableFuture.runAsync(() -> {
//
log.info("runAsync线程执行, 线程名:{}", Thread.currentThread().getName()); // 根据参数查询user对象
final User user = userDAO.oneById(userId);
// 修改age
user.setAge(100);
// 执行update
user.update();
}, threadPoolTaskExecutor); // 执行CompletableFuture, runAsync没有返回, 所以这里使用Void接收
final Void unused = completableFuture.get();
log.info("CompletableFuture线程执行, 更新userId:{}的age成功, 返回结果:{}", userId, unused); /*
再次查询userId对应的user对象
*/
final User user = userDAO.oneById(userId);
// 修改备注
user.setTranmemo("测试CompletableFuture事务问题");
// 执行update
user.update();
log.info("当前上下文, 更新userId:{}的tranmemo成功~", userId); }
}

可以看出,这里的Service使用CompletableFuture开启线程更新了userId对应的age,然后上下文又更新了userId对应的tranmemo,这时候如果没有报错,那么正常的结果就是最后userId对应的age和tranmemo都被修改了。

注意: 这时候Service的方法updateUser上已经添加了事务注解 @Transactional(rollbackFor = Exception.class)

执行代码,结果如下:



对于返回结果,我在项目中实现了ResponseBodyAdvice进行统一返回格式,这里不多说,只要在乎结果执行成功了即可。

数据库查看user表:



很明显,userId为1对应的数据中,age和tranmemo都成功的修改成功了。

2.2.模拟异常情况 -- CompletableFuture执行异常

现在我们假设CompletableFuture中的线程在update之后发生了异常,小小的修改一下代码。

修改后的代码如下:



执行,结果如下:



可以看到,结果执行失败了,异常信息就是我们添加的 int i = 1/0 导致的,现在我们看看数据库中的结果



no~,age发生了改变, 但是tranmemo却没有被修改,这说明啥?是的,CompletableFuture中的线程执行失败后(其实就是 final Void unused = completableFuture.get() 这一行代码的执行失败了,失败后因为没有对异常的处捕获处理,所以就执行失败抛出了异常),后面的上下文代码没有执行了。

但是CompletableFuture中的update却依然生效,这显然不是我们想要的。

这里补充一点,如果CompletableFuture的runAsync方法执行后我们还使用了exceptionally, 结果就会有差异了,我们这里可以试一试。

添加exceptionally后的代码如下:



让我们看看现在执行后的结果:



数据库结果:

看到了嘛!!!age和tranmemo都执行成功了!!!为什么呢?

其实很简单,就是因为添加的exceptionally导致的,还是开头说的,看一看 CompletableFuture使用方法详细说明 后或许你就会明白了(如果你还是不明白,留言就行)。

现在我们继续往下走。

2.3.模拟异常情况 -- 上下文执行异常

现在我们继续假设CompletableFuture中的线程执行成功,但是上下文执行失败的情况。

代码修改后, 如下:



取消了CompletableFuture中的模拟异常,范围在上下文的最后添加模拟异常。

执行结果如下:

执行失败了,这是上下文的异常报出来的,我们再看一下数据库结果:



age被修改了,但是tranmemo没有变化,因为上下文抛出了异常,@transactional起了作用,事务发生了回滚。

进而说明,CompletableFuture中的事务和上下文的事务是分离的,上下文发生回滚的时候,无法回滚CompletableFuture中的修改内容。

显然,这种情况也不是我们想要的。

3.问题解决

针对刚才我们进行的测试,可以总结,无论CompletableFuture还是上下文,只要发生了异常,都存在结果不对的可能,而我们想要的是,无论谁存在问题,结果会回滚。

如何解决问题呢?

那就是手动开启CompletableFuture中的事务,手动回滚!!!

其实可以发现,无论CompletableFuture还是上下文发生异常,我们最终需要控制的,都是CompletableFuture的事务。

3.1.手动回滚异常

好了,现在该来解决刚才出现的问题了,不多说,直接上代码:



代码也贴一遍

@RequiredArgsConstructor
@Slf4j
@Service
public class CompletableFutureTransactionalService { private final ThreadPoolTaskExecutor threadPoolTaskExecutor; private final UserDAO userDAO; private final DataSourceTransactionManager dataSourceTransactionManager; /**
* 根据id修改User, 测试CompletableFuture事务问题
*
* @param userId User主键值
*/
@Transactional(rollbackFor = Exception.class)
public void updateUser(final Integer userId) throws ExecutionException, InterruptedException {
/*
创建CompletableFuture线程
*/
final CompletableFuture<Void> completableFuture = CompletableFuture.runAsync(() -> {
//
log.info("runAsync线程执行, 线程名:{}", Thread.currentThread().getName());
/*
手动开启事务
*/
DefaultTransactionDefinition def = new DefaultTransactionDefinition();
// 事物隔离级别,开启新事务,这样会比较安全些。
def.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRES_NEW);
// 获得事务状态
TransactionStatus status = dataSourceTransactionManager.getTransaction(def); try {
// 根据参数查询user对象
final User user = userDAO.oneById(userId);
// 修改age
user.setAge(666);
// 执行update
user.update(); /*
模拟CompletableFuture异常
*/
int i = 1 / 0; } catch (Exception e) {
// 事务回滚
dataSourceTransactionManager.rollback(status);
throw new RuntimeException("runAsync线程执行失败", e);
} }, threadPoolTaskExecutor); // 执行CompletableFuture, runAsync没有返回, 所以这里使用Void接收
final Void unused = completableFuture.get();
log.info("CompletableFuture线程执行, 更新userId:{}的age成功, 返回结果:{}", userId, unused); /*
再次查询userId对应的user对象
*/
final User user = userDAO.oneById(userId);
// 修改备注
user.setTranmemo("222222");
// 执行update
user.update(); log.info("当前上下文, 更新userId:{}的tranmemo成功~", userId); }
}

我们再次执行:



注意,最然这里执行失败了,但是因为我们在catch中手动抛出了异常,如果不手动抛出异常,上下文就会继续执行,最后执行完修改了tranmemo,这是不对的,因为虽然执行成功了,但实际CompletableFuture是执行失败的(我们回滚了事务)。

再看数据库表:



好了,age和tranmemo都没有改变。

其实万变不离其宗,无论实际业务是怎么样,只要出现这种情况,需要手动控制事务,就这么处理即可(不是百分百这么些,根据实际需要)。

CompletableFuture事务问题的更多相关文章

  1. 006-优化web请求二-应用缓存、异步调用【Future、ListenableFuture、CompletableFuture】、ETag、WebSocket【SockJS、Stomp】

    四.应用缓存 使用spring应用缓存.使用方式:使用@EnableCache注解激活Spring的缓存功能,需要创建一个CacheManager来处理缓存.如使用一个内存缓存示例 package c ...

  2. 【RocketMQ】事务的实现原理

    事务的使用 RocketMQ事务的使用场景 单体架构下的事务 在单体系统的开发过程中,假如某个场景下需要对数据库的多张表进行操作,为了保证数据的一致性,一般会使用事务,将所有的操作全部提交或者在出错的 ...

  3. Spring基于AOP的事务管理

                                  Spring基于AOP的事务管理 事务 事务是一系列动作,这一系列动作综合在一起组成一个完整的工作单元,如果有任何一个动作执行失败,那么事务 ...

  4. SQLServer事务同步下如何收缩日志

    事务同步是SQLServer做读写分离的一种常用的方式. 随着业务数据的不断增长,数据库积攒了大量的日志,为了腾出硬盘空间,需要对数据库日志进行清理 订阅数据库的日志清理 因为订阅数据库所有的数据都来 ...

  5. 事务日志已满,原因为“ACTIVE_TRANSACTION”

    汇总篇:http://www.cnblogs.com/dunitian/p/4822808.html#tsql 异常处理汇总-数据库系列  http://www.cnblogs.com/dunitia ...

  6. Mysql事务探索及其在Django中的实践(二)

    继上一篇<Mysql事务探索及其在Django中的实践(一)>交代完问题的背景和Mysql事务基础后,这一篇主要想介绍一下事务在Django中的使用以及实际应用给我们带来的效率提升. 首先 ...

  7. Mysql事务探索及其在Django中的实践(一)

    前言 很早就有想开始写博客的想法,一方面是对自己近期所学知识的一些总结.沉淀,方便以后对过去的知识进行梳理.追溯,一方面也希望能通过博客来认识更多相同技术圈的朋友.所幸近期通过了博客园的申请,那么今天 ...

  8. CRL快速开发框架系列教程七(使用事务)

    本系列目录 CRL快速开发框架系列教程一(Code First数据表不需再关心) CRL快速开发框架系列教程二(基于Lambda表达式查询) CRL快速开发框架系列教程三(更新数据) CRL快速开发框 ...

  9. 玩转spring boot——结合JPA事务

    接着上篇 一.准备工作 修改pom.xml文件 <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi=&q ...

  10. MySQL 系列(三)你不知道的 视图、触发器、存储过程、函数、事务、索引、语句

    第一篇:MySQL 系列(一) 生产标准线上环境安装配置案例及棘手问题解决 第二篇:MySQL 系列(二) 你不知道的数据库操作 第三篇:MySQL 系列(三)你不知道的 视图.触发器.存储过程.函数 ...

随机推荐

  1. vue中 beforeRouteLeave 生命周期函数

    beforeRouteLeave需求描述在使用 element-UI的table 的时候,有这么一个需求.从一个页面切换到另一个页面,再切回来的时候,滚动条的位置不变. 需求:滚动浏览列表页,出现滚动 ...

  2. ADC-CH32FV2x_V3x 框图功能详解

    主要特性(CH32FV2x_V3x) 系列:l 12 位分辨率l 支持 16 个外部通道和 2 个内部信号源采样l 多通道的多种采样转换方式:单次.连续.扫描.触发.间断等l 数据对齐模式:左对齐.右 ...

  3. .Net6 微服务之Polly入门看这篇就够了

    前言 O(∩_∩)O 大家好!书接上文,本文将会继续建立在 .Net6 使用 Ocelot + Consul 看这篇就够了 项目的基础上进行Polly的介绍,然后这篇文章只是个人学习与分享,不喜勿喷, ...

  4. Activiti01-基本介绍

    1.工作流的定义 工作流是将一组任务组织起来以完成某个有序的过程:定义了任务的触发顺序和触发条件,而且每个任务可以由一个或多个软件系统完成,也可以由一个或一组人完成, 还可以由一个或多个人与软件系统协 ...

  5. 下午小博(java小知识)

    抽象类: 抽象类中可以构造方法 抽象类中可以存在普通属性,方法,静态属性和方法 抽象类中可以存在抽象方法如果一个类中有一个抽象方法,那么当前类一定是抽象类:抽象类中不一定有抽象方法 抽象类中的抽象方法 ...

  6. SRE:如何提高报警有效性?

    为什么要提升<报警有效性> 过多的报警会让负责人麻木 过多的报警会增加短信和电话的成本 提升根因定位效率 如何定义<报警有效性> 不漏报 不误报 不重报 不延报 如何量化 MT ...

  7. Centos8安装nvidia驱动

    Centos8安装nvidia驱动 1. 查看显卡型号 lspci | grep-i nvidia 或者 lspci -vnn | grep VGA 2. 前往nvidia官网下载对应驱动 NVIDI ...

  8. 分布式共识算法随笔 —— 从 Quorum 到 Paxos

    分布式共识算法随笔 -- 从 Quorum 到 Paxos 本文主要参考各类英文文献,部分专业术语翻译较为生硬,望谅解. 概览: 为什么需要共识算法? 昨夜西风凋碧树,独上高楼,望尽天涯路 复制(Re ...

  9. JZOJ 2022.01.21【提高A组】模拟

    简要题解加心得 不得不说这是我打得比较痛苦且改得比较痛苦的一套题了 \(\text{T1 1085. [GDOI2008]彩球游戏}\) 整整改了三个半小时 直接崩溃了 明明本地可以跑过去,偏偏 \( ...

  10. java语言——跨平台原理,jre,jdk

    day1 Java是一种混合的编译运行方式:编译+解释(虚拟机) java的跨平台:在虚拟机中运行(jvm) jdk:jvm,核心类库,开发工具(开发环境) jre:Java的运行环境