@Transactional注解:多个事务嵌套时,独立事务处理方式

@Transactional注解:多事务嵌套,独立事务处理

看下需求

在多个事务嵌套使用时,排除事务之间的回滚影响

解决方案

在不同服务类的方法中使用Transactional的propagation属性来实现隔离事务。(注意两个方法不在同一个服务类中)

Propagation.REQUIRES_NEW即说明该事务开启单独事务,不受其他事务影响

// 服务类A
@Autowired
private ABizService aBizService;
 
@Override
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public String insert(FacedbInfoDTO facedbInfoDTO) throws ServiceException {
    ... 
    aBizService.generateId();
} 
 
// 服务类B
@Override
@Transactional(propagation = Propagation.REQUIRES_NEW, rollbackFor = Exception.class)
public String generateId() {
    ...    
}

嵌套事务分析@Transactional

事务类型总共有七种;在这就介绍常用的的两个。 

@Transactional(propagation=Propagation.REQUIRED)//如果有事务,那么加入事务,没有的话新创建一个;不指定propagation默认就是这个
@Transactional(propagation=Propagation.REQUIREDS_NEW)//不管是否存在事务,都创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务,

Propagation.REQUIRED类型事务嵌套

A事务方法调用B事务方法时如果两个事务注解在生效的情况下,在这里A事务称为父类,B事务称为子类;A方法操作数据库后调用B方法,下列异常抛出时保证AB方法里对数据库操作都完成了才抛出异常;RuleException继承的RuntimeException;

列举一下测试结果:

@Transactional注解:多个事务嵌套时,独立事务处理方式_第1张图片

总结:

1.如果子类方法抛出的异常,不管满足子类还是父类的注解的回滚事务就会回滚;

2.如果父类抛出异常,只看父类的事务注解,如果回滚就父子皆回滚,如果不回滚父子皆不回滚

嵌套事务类型是Propagation.REQUIRES_NEW

这种情况下子事务开启了新的session,父事务先锁定一条记录(for update)子事务也锁这一条记录时就会死锁;这点要注意;

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

你可能感兴趣的:(@Transactional注解:多个事务嵌套时,独立事务处理方式)