场景:
使用Sping Aop控制事务,ServiceA调用ServiceB,如果ServiceB中抛出异常,抛出的为业务异常(自定义异常),虽然在ServiceA中已经捕捉处理了ServiceB中抛出的异常,但是ServiceA执行结束后,报如下异常:
org.springframework.transaction.UnexpectedRollbackException:Transaction rolled back because it has been marked as rollback-only
原因:
在调用ServiceB时,ServiceB中抛出了异常,Spring事务会将其标记为回滚状态,而不是直接回滚,虽然ServcieA捕获了这个自定义异常,但在ServiceA执行结束后,因已标记为回滚状态,则整体回滚,抛出以上异常。
解决:
方式一:
将ServiceB中使用的业务自定义异常废除,改用标记的形式进行返回,告知ServiceA,ServiceB的执行情况。
如果改动不便,直接在ServiceA中对逻辑进行判断,避免传参至ServiceB中引起自定义异常的抛出而导致整体回滚。
方式二:
在spring事务中配置globalRollbackOnParticipationFailure为false,表示如果遇到部分失败,整个事务不会回滚。但是这会影响其他业务逻辑的事务
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
<property name="globalRollbackOnParticipationFailure" value="false" />
</bean>