从 rollback-only异常中分析下事务中propagation的应用和理解
rellback-only异常场景描述
项目已经过去很久了,今天突然间出现一个BUG,我开发的接口自己这边运行很正常,但是别人调用的时候,在方法结束后进行事务提交时报错:
Transaction rolled back because it has been marked as rollback-only
中文翻译就是:
事务已回滚,因为它被标记成了只回滚
以下是我的代码和调用我的代码(剥离业务,直接将代码提取出来)
// 我的代码
@Transactional(readOnly = true, rollbackFor = Exception.class)
public String select(String pid,String id) {
List<Object> list = this.commonService.select(pid,id);
if (CollectionUtils.isEmpty(list)) {
throw new ApplicationException("查询不到数据");
}
...
}
.....
// 调用方的代码
@Transactional(readOnly = true, rollbackFor = Exception.class)
public String select(String pid,String id) {
try{
// 针对异常进行捕获,但是没有抛出异常
this.measureService.select(pid,id);
}catch(ApplicationException e) {
}
}
错误的原因经过分析是:
A和B都有事务,A调用B,B中抛出了异常,A这边捕获B的异常,但是没有将异常抛出,导致A方法执行结束时,提交事务,出现了上述的错误
从上面的代码我们可以看出,是一个嵌套事务,那么出现上面异常的主要原理是什么呢?
首先我们理解基于注解@Transactional,其内部是基于Aop实现的:
SpringAop异常捕获的原理:本拦截的方法需显示抛出异常,并不能经过任何处理,这样Aop代理才能捕获方法的异常,才能进行回滚,默认情况下Aop只捕获RuntimeException异常
这样原理就简单了,我们知道事务是可以继承的,那么A和B中可以理解为同一个事务,我们在B中进行了抛出了异常,然后再A中进行了捕获,但是没有抛出该异常,相当于在A中进行了处理,没有抛出该异常。细分下:B方法返回时,transcation已经被设置为rollback-only了,但是A这边捕获异常,没有继续向外抛,那么A方法结束时,就会由Aop来提交事务,但是此时transcation已经被设置为rollback-only了,所以就会抛出上述异常。
Rollback-only异常的解决
既然我们知道原理了,那么我们就来解决上述的问题:
- B方法中我们可以手动捕获异常,并且处理该异常
try {
int i = 1/0;
}catch (Exception e) {
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
}
- B方法中的事务我们可以开启一个新的事务,与A中的事务不一样,这样A中出现异常回滚的事务与B中的事务不是同一个事务,这样话就可以避免上述的问题了,但是这样会出现一个问题,如果B中也有对数据的操作,那么B中出现异常,是不会让A中的事务回滚的。该方法适用于B中没有对数据的操作
// 在B中方法上添加注解,另开一个事务
@Transactional(propagation = Propagation.REQUIRES_NEW)
网友评论