java等待事务提交后回调另一个异步方法

问题复现

1.场景

2个service方法, 方法A中调用方法B。

方法A 是核心业务方法,涉及多张表数据变更,为了保持数据一致,用spring事务注解:@Transactional(rollbackFor = Exception.class)

方法B 比较耗时,为了不影响核心业务,方法B 用@Async注解,单独开启一个线程去异步执行。(方法B在另外一个类里边,不能和A在同一个类)。

2.出错原因

方法B是异步方法,导致方法A事务还没提交时(不一定出错,具体就看哪个线程执行的快了)方法B就执行了。

3.期望

期望方法A上的大事务commit后再执行方法B。

二、解决方案

// 注册事务同步处理
TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronizationAdapter() {
@Override
public void afterCommit() {
    // 事务提交完毕时,触发:funcB
    funB()}

三、原理

提交一个事务同步处理,在事务commit之后执行,具体存放在threadLocal(线程本地变量)中,事务commit时会去threadLocal里边取。源码afterCommit是空的,没有任何操作,可见是spring专门预留给大家使用的。

源码:

TransactionSynchronizationAdapter是一个接口适配器,这样不用实现接口的全部方法,按需Override即可。类图如下图所示:
在这里插入图片描述
TransactionSynchronizationAdapter实现了2个接口:

TransactionSynchronization事务同步接口,
Ordered执行优先级
我们这里就是实现了TransactionSynchronization接口的afterCommit()方法,最终在事务commit提交后执行。
关于spring事务执行过程图:
在这里插入图片描述

四、总结

遇到问题后,很快就想到了处理方式,因为我提前储备了相关知识:

1.spring事务系列(具体在第三章 事务源码,里边有链接)
spring事务详解(一)初探事务

2.@Async实现异步
异步任务spring @Async注解源码解析

3.threadLocal线程本地变量
ThreadLocal终极源码剖析-一篇足矣!

注:参考原文地址:@Transactional事务提交后触发异步方法

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 我可以为你提供一些关于如何使用Java实现一个异步方法的帮助。首先,你需要使用Java的Future接口来定义一个异步方法,这样可以确保你的方法可以在其他线程中运行,而不会阻塞主线程。其次,你需要使用Java的ExecutorService来启动一个新的线程,并将你的异步方法提交到这个新线程中,以便可以在另一个线程中执行它。最后,你需要使用Future接口来获取异步方法的返回值,以便可以继续处理结果。 ### 回答2: Java中实现异步方法有多种方式,以下是其中两种常见的方式: 1. 使用线程池: 可以使用线程池来执行异步方法,具体步骤如下: 1)创建一个线程池对象,可以使用Java提供的ThreadPoolExecutor类。 2)定义一个Runnable或Callable接口的实现类,实现异步方法的逻辑。 3)将该实现类对象提交到线程池中执行。 4)可以通过Future对象获取异步方法的执行结果。 2. 使用CompletableFuture: Java 8引入了CompletableFuture类,用于实现简单的异步编程,具体步骤如下: 1)使用CompletableFuture.supplyAsync()或CompletableFuture.runAsync()方法创建一个异步执行的任务。 2)在异步任务中定义异步方法的逻辑,可以使用lambda表达式。 3)通过调用CompletableFuture对象的join()、get()或join(long timeout, TimeUnit unit)方法获取异步方法的执行结果。 这两种方式都可以实现异步方法,具体选择哪种方式取决于具体的需求和开发场景。注意在使用异步方法时要注意线程安全性和异常处理,避免出现多线程竞争或异常问题。同时,异步方法的调用方需要注意合理处理异步方法的执行结果,避免出现程序阻塞或异常无法处理的情况。 ### 回答3: 要实现一个异步方法,可以采用Java提供的两种方式:使用线程池或使用CompletableFuture。 第一种方式是使用线程池。可以创建一个线程池,通过submit方法提交任务,然后通过Future对象获取异步结果。具体的步骤如下: 1. 创建一个线程池,可以使用ExecutorService的实现类ThreadPoolExecutor。 2. 定义一个方法,该方法用于执行异步操作。在方法内部,可以使用submit方法提交一个Callable对象,该Callable对象包含要执行的异步任务。 3. 获取异步任务的结果,可以通过Future对象的get方法获取。如果异步任务还没有完成,get方法会阻塞直到任务完成并返回结果。 第二种方式是使用CompletableFuture。这是Java 8新增的一个类,用于实现异步编程。下面是实现异步方法的步骤: 1. 定义一个方法,该方法用于执行异步操作。在方法内部,可以创建一个CompletableFuture对象。 2. 在CompletableFuture对象上调用supplyAsync方法,并传入要执行的异步任务,这个任务将会在另一个线程中执行。 3. 在supplyAsync方法的返回值上调用thenApply方法,传入一个函数,用于处理异步任务的结果,并返回处理后的结果。 4. 调用CompletableFuture对象的get方法获取异步任务的返回值。get方法会阻塞直到任务完成并返回结果。 这两种方式都可以实现异步方法,具体选择哪一种取决于实际需求和个人喜好。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值