@Transactional 事务提交之后执行 @Async 修饰的异步方法

最近项目中遇到的问题:

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

方法A核心业务涉及多张表的数据操作,事务采用注解:@Transactional(rollbackFor = Exception.class)。

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

出现的问题:

方法A的事务还没提交,方法B就执行了,导致方法B中查到的数据还是老数据。

当时想到的解决方案,方法A事务提交后再执行方法B。

问题代码:

class A {
 
    @Autowired
    private B b;
    
    @Transactional
    public void updateA(..) {
        insert(..);
        update(..);
        b.updateB(..);
    }
 
}
    

class B {
 
    @Async
    public void updateB(..) {
        update(..)
    }
 
}

分析
        方法A 和 方法B 假如在同一个类中,则方法B 的 @Async 注解会失效:
        首先解释下 @Transactional 注解失效的原因:Spring 在扫描 bean 的时候会扫描方法上是否包含 @Transactional 注解,如果包含,Spring 会为这个 bean 动态地生成一个子类(即代理类 proxy),代理类是继承原来那个 bean 的。此时,当这个有注解的方法被调用的时候,实际上是由代理类来调用的,代理类在调用之前就会启动 Transaction。然而,如果这个有注解的方法是被同一个类中的其他方法调用的,那么该方法的调用并没有通过代理类,而是直接通过原来的那个bean,所以就不会启动 Transaction,我们看到的现象就是 @Transactional 注解无效。

        因为 @Transactional 和 @Async 注解的实现都是基于 Spring 的 AOP ,而 AOP 的实现是基于动态代理实现的,故Async 失效的原理和原理是一样的。(排除配置错误导致失效的原因)

        常用的解决办法是将调用方法和被调用方法(@Transactional 或 @Async 修饰的方法)放到两个类中,保证通过代理类去调用被 @Transactional 或 @Async 修饰的方法。

        如何保证方法A 的内部方法 insert(..) 和 update(..) 事务提交之后再异步执行方法B?
        通过以下代码实现:

TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronizationAdapter() {
@Override
public void afterCommit() {
    // 事务提交完毕时,触发:方法B
    updateB(..);
}

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

解决
        最终实现:

class A {
 
    @Autowired
    private B b;
    
    @Transactional
    public void updateA(..) {
        insert(..);
        update(..);
        TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronizationAdapter() {
            @Override
            public void afterCommit() {
                b.updateB(..);
            }
        });
    }
 
}
    
class B {
 
    @Async
    public void updateB(..) {
        update(..)
    }
 
}

🥳

  • 8
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值