问题描述
serviceA中的methodA调用serviceB中的methodB,methodB抛出一个异常,然后methodA捕获掉这个异常,就会出现
Transaction rolled back because it has been marked as rollback-only 异常
(serviceA和serviceB都加了注解@Transactional(rollbackFor = Exception.class))
问题出现的环境背景及自己尝试过哪些方法
业务需求是在serviceA中的methodA中调用serviceB中的methodB,但是methodB有一定几率会抛异常,methodA要把异常捕获,并做其他处理。
但是methodA返回结果的时候就会报“Transaction rolled back because it has been marked as rollback-only”异常。
解决办法是在methodB上添加注解@Transactional(rollbackFor = Exception.class, propagation = Propagation.REQUIRES_NEW)。还有一种办法是把methodB写到serviceA中,直接用this.methodB调用
疑惑
现在想搞明白为什么serviceA不能捕获serviceB中的异常?
serviceA中的methodA中调用serviceB中的methodB和同一个service中两个方法调用,在事务处理上有什么区别?
不是不能捕获,你在
serviceA
中确实捕获了serviceB
中的异常,只是异常捕获处理之后事务进行提交之前进行了检查从而又抛出了异常,简单分析下题主的场景:因为
spring
的默认传播属性,serviceB
的事务加入到serviceA
的事务中,二者在同一个事务中,假设在serviceB
中发生了异常(比如字段长度100而内容长度150),你在serviceA
中捕获了该异常,所以事务没有进行回滚,当serviceA
的事务进行提交之前,因为serviceB
发生过SQL异常
,导致整个事务标记为不可提交状态(通过serviceB
的代理方法标记),所以重新抛出Transaction rolled back because it has been marked as rollback-only
异常。SQL
本来就是错的当然不能提交咯。当你把传播属性改为Propagation.REQUIRES_NEW
,两者就不在同一事物内了,也就是说serviceB
的错误sql
不会影响serviceA
的事务。第二个问题的答案,你看看这个https://segmentfault.com/q/10...