spring报错→UnexpectedRollbackException: Transaction silently rolled back becaus

1、前言

  考虑到功能的划分,将一个功能模块的不同职责划分为到几个service中,同时为service添加数据库事务(即添加@Service、@Transactional注解,添加@Transactional注解到类上,则类中的方法都会添加事务),由于需要中途报错不能直接返回,需要完全执行完再把结果返回,故添加try-catch语法糖,try-catch里面调用其他service 方法,解决过程涉及到spring 事务的传递性,下面再做讲解。

2、报错重现

Test2代码:

@Service
@Transactional
public class Test2 {
    public void test2(){
        throw new RuntimeException("test2 throw");
    }
}

Test1代码:

@Service
@Transactional
public class Test1 {
    @Autowired
    private Test2 test2;
    public void test1(){
        try {
            test2.test2();
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

junit调用代码:

@RunWith(SpringRunner.class)
@SpringBootTest
public class TestTransaction {
    @Autowired
    private Test1 test1;
    @Test
    public void transaction() {
        test1.test1();
    }
}

这里重点是Test1实例方法try-catch了test2的报错。

3、spring事务的传播性及分析

事务传播行为类型:

事务传播行为类型 说明
PROPAGATION_REQUIRED 如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中。这是最常见的选择。
PROPAGATION_SUPPORTS 支持当前事务,如果当前没有事务,就以非事务方式执行。
PROPAGATION_MANDATORY 使用当前的事务,如果当前没有事务,就抛出异常。
PROPAGATION_REQUIRES_NEW 新建事务,如果当前存在事务,把当前事务挂起。
PROPAGATION_NOT_SUPPORTED 以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
PROPAGATION_NEVER 以非事务方式执行,如果当前存在事务,则抛出异常。
PROPAGATION_NESTED 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与PROPAGATION_REQUIRED类似的操作。

表格引用自:Spring事务传播机制

spring报错→UnexpectedRollbackException: Transaction silently rolled back becaus_第1张图片
  从注解的默认值可以看到,PROPAGATION_REQUIRED是默认值,那么就要想一下,如果报错重现一样出现事务嵌套,使用默认值会出现怎样的情况。
  PROPAGATION_REQUIRED传播性描述是这样的:

如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中。这是最常见的选择

所以我们描述一下过程:
1、test1方法开启事务
2、调用Test2类实例的test2方法,由于事务传播特性都是默认值,所以test2方法根据PROPAGATION_REQUIRED的特性,会将本身事务加入到test1的事务中,就是用沿用test1的事务。
3、test2加入test1方法的事务
4、test2抛出异常
5、test1进行try-catch
此时test2抛出异常,会导致整个事务回滚,即test1和test2和合并事务回滚,test1本身的不想报错就回滚的操作的,是想让所有方法执行完。但属于同一个事务,所以导致报错:

org.springframework.transaction.UnexpectedRollbackException: Transaction silently rolled back because it has been marked as rollback-only

解决方法是调用Test2实例的test2方法的时候,新建一个单独事务。
1、符合要求的是PROPAGATION_REQUIRES_NEW(新建事务,如果当前存在事务,把当前事务挂起。
2、试了一下PROPAGATION_NESTED(如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与PROPAGATION_REQUIRED类似的操作。)这个特性,好像报了另一个错误。

解决方法是给子事务更改事务传播特性,Test2改为:

@Service
@Transactional
public class Test2 {
    @Transactional(propagation = Propagation.REQUIRES_NEW)//这是新增
    public void test2(){
        throw new RuntimeException("test2 throw");
    }
}

你可能感兴趣的:(java)