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

目录

  • @Transactional注解:多事务嵌套,独立事务处理
    • 看下需求
    • 解决方案
  • 嵌套事务分析@Transactional
    • Propagation.REQUIRED类型事务嵌套
    • 嵌套事务类型是Propagation.REQUIRES_NEW

@Transactional注解:多事务嵌套,独立事务处理
看下需求
在多个事务嵌套使用时,排除事务之间的回滚影响

解决方案
在不同服务类的方法中使用Transactional的propagation属性来实现隔离事务。(注意两个方法不在同一个服务类中)
Propagation.REQUIRES_NEW即说明该事务开启单独事务,不受其他事务影响
// 服务类A@Autowiredprivate 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注解:多个事务嵌套时,独立事务处理方式】
嵌套事务分析@Transactional 事务类型总共有七种;在这就介绍常用的的两个。
@Transactional(propagation=Propagation.REQUIRED)//如果有事务,那么加入事务,没有的话新创建一个; 不指定propagation默认就是这个@Transactional(propagation=Propagation.REQUIREDS_NEW)//不管是否存在事务,都创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务,


Propagation.REQUIRED类型事务嵌套
A事务方法调用B事务方法时如果两个事务注解在生效的情况下,在这里A事务称为父类,B事务称为子类;A方法操作数据库后调用B方法,下列异常抛出时保证AB方法里对数据库操作都完成了才抛出异常;RuleException继承的RuntimeException;
列举一下测试结果:
@Transactional注解:多个事务嵌套时|@Transactional注解:多个事务嵌套时,独立事务处理方式
文章图片

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

嵌套事务类型是Propagation.REQUIRES_NEW
这种情况下子事务开启了新的session,父事务先锁定一条记录(for update)子事务也锁这一条记录时就会死锁;这点要注意;
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

    推荐阅读