更深入了解 Spring 事务。
客户注册完成后,需要给该客户登记一门PUA必修课,并升级该门课的登记客户数。
为此,我增加了两个表。
课程表 course,记录课程名称和注册的客户数。
客户选课表 user_course,记录客户表 user 和课程表 course 之间的多对多关联。
新添加客户选课记录
课程登记学生数 + 1
新添加业务类 CourseService实现相关业务逻辑,分别调用了上述方法保存客户与课程的关联关系,并给课程注册人数+1
为避免注册课程的业务异常导致客户信息无法保存,这里 catch 注册课程方法中抛出的异常。希望当注册课程发生错误时,只回滚注册课程部分,保证客户信息仍然正常。
执行代码:
伪代码梳理整个事务的结构:
整个业务包含2层事务:
Spring公告式事务中的propagation属性,表示对这些方法使用怎么的事务,即:
一个带事务的方法调用了另一个带事务的方法,被调用的方法它怎样解决自己事务和调用方法事务之间的关系。
propagation 有7种配置:
由于:
regCourse() 就会加入到已有的事务中,两个方法共用一个事务。
Spring 事务解决的核心:
protected Object invokeWithinTransaction(Method method, @Nullable Class<?> targetClass, final InvocationCallback invocation) throws Throwable { TransactionAttributeSource tas = getTransactionAttributeSource(); final TransactionAttribute txAttr = (tas != null ? tas.getTransactionAttribute(method, targetClass) : null); final PlatformTransactionManager tm = determineTransactionManager(txAttr); final String joinpointIdentification = methodIdentification(method, targetClass, txAttr); if (txAttr == null || !(tm instanceof CallbackPreferringPlatformTransactionManager)) { // 能否需要创立一个事务 TransactionInfo txInfo = createTransactionIfNecessary(tm, txAttr, joinpointIdentification); Object retVal = null; try { // 调用具体的业务方法 retVal = invocation.proceedWithInvocation(); } catch (Throwable ex) { // 当发生异常时进行解决 completeTransactionAfterThrowing(txInfo, ex); throw ex; } finally { cleanupTransactionInfo(txInfo); } // 正常返回时提交事务 commitTransactionAfterReturning(txInfo); return retVal; } //......省略非关键代码.....}
整个方法完成了事务的一整套解决逻辑,如下:
当前案例是两个事务嵌套,外层事务 saveUser()和内层事务 regCourse(),每个事务都会调用到这个方法。所以,该方法会被调两次。
当捕获了异常,会调用
进行异常解决:
对异常类型做了少量检查,当符合公告中的定义后,执行具体的 rollback 操作,这个操作是通过如下方法完成:
该回滚实现负责解决正参加到已有事务集的事务。委托执行Rollback和doSetRollbackOnly。
继续调用
该方法里区分了三种场景:
由于默认传播类型REQUIRED,嵌套的事务并未开启一个新事务,所以属于当前事务处于一个更大事务中,所以会走到分支1。
如下的判断条件确定能否设置为仅回滚:
if (status.isLocalRollbackOnly() || isGlobalRollbackOnParticipationFailure())
满足任一,都会执行 doSetRollbackOnly():
isLocalRollbackOnly
默认 false,当前场景为 false
isGlobalRollbackOnParticipationFailure()
所以,就只由该方法来确定了,默认值为 true, 即能否回滚交由外层事务统一决定
条件得到满足,执行
最终调用
内层事务操作执行完毕。
外层事务中,业务代码就捕获了内层所抛异常,所以该异常不会继续往上抛,最后的事务会在 TransactionAspectSupport.invokeWithinTransaction()
中的
该方法里执行了commit 操作:
当满足 !shouldCommitOnGlobalRollbackOnly() &&defStatus.isGlobalRollbackOnly()
,就会回滚,否则继续提交事务:
isGlobalRollbackOnly()
该方法最终进入
之前内部事务解决最终调用到DataSourceTransactionObject#setRollbackOnly()
public void setRollbackOnly() { getConnectionHolder().setRollbackOnly();}
两个方法本质都是对ConnectionHolder.rollbackOnly
属性标志位的存取
但ConnectionHolder则存在于DefaultTransactionStatus#transaction属性。
综上:外层事务能否回滚的关键,最终取决于DataSourceTransactionObject#isRollbackOnly(),该方法返回值正是在内层异常时设置的。
所以最终外层事务也被回滚,从而在控制台中打印上述日志。
这就明白了,Spring默认事务传播属性为REQUIRED:若已有事务,则加入该事务,若无事务,则创立新事务,因此内外两层事务都处于同一事务。
在 regCourse()中抛异常,并触发回滚操作时,这个回滚会继续传播,从而把 saveUser() 也回滚,最终整个事务都被回滚!
Spring事务默认传播属性 REQUIRED,在整个事务的调用链上,任一环节抛异常都会导致全局回滚。
所以只要将传播属性改成 REQUIRES_NEW :
TransactionAspectSupport.invokeWithinTransaction()
中调用 createTransactionIfNecessary()
就会创立一个新的事务,独立于外层事务