@Transactional如何在事务方法中正确开启异步任务?

需求

用户注册完成以后,异步给用户发送邮件,代码如下:

// 用户注册的controller入口
@GetMapping(value="/register")
public int register() {
	return userService.register();
}
// 用户注册的service
@Transactional(rollbackFor = Exception.class)
public int register() {
    // 注册用户
    User emp = new User(id.incrementAndGet(), "若鱼1919", 1);
    int result = userMapper.insert(emp);
    // 为了不阻塞主线程,异步发送邮件
    emailService.sendEmail(emp.getId());
    // 模拟代码执行受塞
    try {
        Thread.sleep(1000);
    } catch (InterruptedException e) {
        throw new RuntimeException(e);
    }
    return result;
}
// emailService
@Async
public void sendEmail(int userId) {
    User user = userMapper.selectById(userId);
    if(user == null){
        throw new RuntimeException("user信息不存在");
    }
    logger.info("向id是{}的用户{}发送邮件成功", userId, user.getName());
}

用户注册的方法上添加了@Transactional注解用于事务控制,用户注册完成以后,为了不阻塞主线程,使用异步的方式向用户发邮件,同时在用户注册方法的最后加了一个Thread.sleep(1000);模拟代码执行受阻的情形。这个时候因为发邮件是异步执行的,因此会先于注册方法执行,此时事务还没提交,因此根据id查询不到数据,导致抛出异常:user信息不存在。

解决办法

1)在异步方法中sleep一下

这种方式最简单,但是sleep的时间不好控制,不推荐

2)编程式事务处理

这种方式可行,但是,代码的侵入性太高,不推荐

public int register() {
    User emp = null;
    int result = 0;
    // 手动开启事务
    TransactionDefinition transactionDefinition = new DefaultTransactionDefinition();
    TransactionStatus transaction = txManager.getTransaction(transactionDefinition);
    try{
        // 注册用户
       emp = new User(id.incrementAndGet(), "若鱼1919", 1);
       result = userMapper.insert(emp);
       // 提交事务
       txManager.commit(transaction);
    }catch(Exception e){
       // 回滚事务 
       txManager.rollback(transaction);
       throw new RuntimeException(e);
    }
    // 异步发送邮件
    emailService.sendEmail(emp.getId());
    // 模拟代码执行受塞
    try {
        Thread.sleep(1000);
    } catch (InterruptedException e) {
        throw new RuntimeException(e);
    }
    return result;
}

因为在发送邮件之前,已经把事务提交掉了,因此不会出现查不到数据的情况。

3)使用事务事件监听器(推荐)

首先定义一个事件,注册成功以后,把这个事件发送出来:

public class UserRegisterEvent extends ApplicationEvent {
    private Integer userId;
    public UserRegisterEvent(Object source, Integer userId) {
        super(source);
        this.userId = userId;
    }
    public Integer getUserId() {
        return userId;
    }
}

用户注册的逻辑还是使用声明式事务注解@Transactional,但是注册完成以后,改成发UserRegisterEvent 消息:

@Transactional(rollbackFor = Exception.class)
public int register() {
    // 注册用户
    User emp  = new User(id.incrementAndGet(), "若鱼1919", 1);
    int result  = userMapper.insert(emp);
    // 异步发送邮件
    applicationContext.publishEvent(new UserRegisterEvent(this, emp.getId()));
    //emailService.sendEmail(emp.getId());
    // 模拟代码执行受塞
    try {
        Thread.sleep(1000);
    } catch (InterruptedException e) {
        throw new RuntimeException(e);
    }
    return result;
}

然后写一个监听器,监听这个事件:

@Async
@TransactionalEventListener(phase = TransactionPhase.AFTER_COMMIT, classes = UserRegisterEvent.class)
public void onRegisterEvent(UserRegisterEvent event) {
    sendEmail(event.getUserId());
}

需要在监听的处理方法上添加@TransactionalEventListener注解,phase的取值有以下几种:

public enum TransactionPhase {
    BEFORE_COMMIT, //事务提交之前
    AFTER_COMMIT,  // 事务提交以后
    AFTER_ROLLBACK, // 事务回滚以后
    AFTER_COMPLETION; // 事务执行完以后
    private TransactionPhase() {
    }
}

以上代码的含义就是当事务提交以后会回调到这个监听方法中,因为事务已经提交,此时是可以正常查询到数据的。
既没有代码的侵入,也不需要担心sleep时间太久,推荐!
源码下载

`try-catch` 和 `@Transactional` 是两种不同的机制,用于处理异常和数据库事务管理,通常在Java的Spring框架中使用。它们在某些情况下可能会失效或不协同工作,这通常发生在以下情况: 1. **嵌套事务**:如果在一个`@Transactional`方法内部又有一个`try-catch`块,Spring默认不会自动开始另一个事务,所以`catch`部分可能不会在事务上下文中执行。要解决这个问题,你需要显式地开启一个新的事务。 ```java @Transactional(propagation = Propagation.REQUIRES_NEW) try { // ... } catch (Exception e) { // 异常处理 } ``` 2. **事务回滚策略**:`@Transactional`注解提供了回滚规则(如`rollbackFor`),如果异常匹配指定的条件,事务才会回滚。如果`try-catch`中的异常类型不满足这个规则,`catch`部分的代码可能在已经完成的事务中运行,导致数据状态不一致。 3. **异常传播**:`@Transactional`中的`throwable`属性定义了异常如何在事务边界间传播。如果`catch`块抛出了一个未处理的异常,这可能会影响`@Transactional`行为,特别是当`propagation`设置为`REQUIRED`时。 4. **异步处理**:如果`catch`中的代码包含异步任务(如发送邮件或调用远程服务),由于这些操作可能不在Spring的事务管理范围之内,`@Transactional`的效果可能会受限。 为了确保`try-catch`与`@Transactional`的正确协同,建议在`catch`块中明确处理事务(如显式回滚或关闭事务)并考虑异常传播规则。同时,如果遇到问题,确保检查你的配置和代码逻辑是否正确使用了这两个特性。如果有疑问,可以详细查看Spring文档或使用Spring提供的AOP(面向切面编程)工具来调试。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值