@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;
列举一下测试结果:
总结:
1.如果子类方法抛出的异常,不管满足子类还是父类的注解的回滚事务就会回滚;
2.如果父类抛出异常,只看父类的事务注解,如果回滚就父子皆回滚,如果不回滚父子皆不回滚
嵌套事务类型是Propagation.REQUIRES_NEW
这种情况下子事务开启了新的session,父事务先锁定一条记录(for update)子事务也锁这一条记录时就会死锁;这点要注意;
以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。