业务场景:
我们经常会有些方法会调用一些方法异步执行,比如入库操作后要异步发送kafka消息,
但是往往我们的业务方法总是包含事务的,要么全部成功,要么全部失败。
有的时候事务执行失败了或者还没有提交,异步方法就执行完成了。
在异步方法中需要查询新插入的数据,发现查询不到,原因是事务未提交,数据库插入数据操作并未完成,只有处在同一个事务中才能查询到新数据,异步方法已经不处于同一个事务中查询表是查不到数据的。
业务要求:
1.异步操作需要等待事务完成后才执行。
2.当事务失败回滚时,我们的异步操作也不执行。
实现方案:
使用TransactionSynchronizationManager在事务提交之后操作
@Transactional()
public void save() {
User user = new User();
user.setName("笑笑");
user.setSex("男");
userMapper.insert(user);
//如果处于事务中,待事务提交成功了再执行后续操作,防止事务未提交完成导致后续操作查不到最新数据
if(TransactionSynchronizationManager.isActualTransactionActive()){
TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronization() {
@Override
public void afterCommit() {
sendTopic();
}
});
}else {
sendTopic();
}
}
isActualTransactionActive方法用来检测是否处于事务中
添加了@Transactional注解,返回true
@Test
@Transactional
public void test1() {
assertTrue(TransactionSynchronizationManager.isActualTransactionActive());
}
当删除 @Transactional 注解时,返回 false:
@Test
public void test2() {
assertFalse(TransactionSynchronizationManager.isActualTransactionActive());
}