开心一刻
昨晚,小妹跟我妈聊天
小妹:妈,跟你商量个事,我想换车,资助我点呀
妈:哎呀,你那分扣的攒一堆都够考清华的,还换车资助点,有车开就不错了
小妹:你要是这么逼我,别说哪天我去学人家傍大款啊
妈:哎呀妈,你脸上那褶子比你人生规划都清晰,咋地,大款缺地图呀,找你?
小妹:让我回到我18岁,大个、水灵、白,你再看看
妈:你18长的像黑鱼棒似的,还水灵白,消防栓水灵,也没见谁娶它呀,女人呐,你得有内涵
data:image/s3,"s3://crabby-images/37818/37818fc1d88d2ee9ac9beb3497ccb91ad3d4e1cc" alt=""
前情回顾
我们知道了女神偶尔的消息可能是借钱
那你到底是借还是不借?
data:image/s3,"s3://crabby-images/7f62f/7f62f8fb5c2aff681374a80fca725413140ad43e" alt=""
不好意思,貌似抓错重点了
重点应该是:把消息发送从事务中拎出来就好了,也就是等事务提交后,再发消息
什么,没看记一次线上问题 → 偶尔的热情真的难顶呀!,不知道重点,那还不赶紧去看?
我光提了重点,但是没给你们具体实现,就问你们气不气?
data:image/s3,"s3://crabby-images/30ccb/30ccb593b440a30380370d009e1def3cff235c40" alt=""
本着认真负责的态度,我还是提供几种实现,谁让我太宠你们了
事务拎出来
说起来很简单,做起来其实也很简单
犯病拎
为了更接近真实案例,我把
data:image/s3,"s3://crabby-images/fec44/fec448d61b0e97a4fb0213524628f24b79d7c1dc" alt=""
调整一下
data:image/s3,"s3://crabby-images/52f89/52f896d4e33255314890f9210d8027c15ffd2b70" alt=""
User更新 和 插入操作日志 在一个事务中, 发消息 需要拎出去
拎出去还不简单,看我表演
data:image/s3,"s3://crabby-images/f2fe5/f2fe5f7ac93f39bbb718f8758d58eea11a914da0" alt=""
相信大家都能看懂如上代码,上游调用 update 的地方也不用改,简直完美!
data:image/s3,"s3://crabby-images/72119/72119c1a2e52ec84e6a0ab7b2a2da5a08df50e9d" alt=""
大家看仔细了, update 上的 @Transactional(rollbackFor = Exception.class) 被拿掉了,不是漏写了!
如果 update 上继续保留 @Transactional(rollbackFor = Exception.class)
data:image/s3,"s3://crabby-images/b36fb/b36fb71a574903c695e148b7343dfa7b2c5859f1" alt=""
是什么情况?
那不是和没拎出来一样了吗?特么的还多写了几行代码!
回到刚拎出来的情况, update 和 updateUser 在同一个类中,非事务方法 update 调用了事务方法 updateUser ,事务会怎么样?
如果你还没反应过来,八股文需要再背一背了:在同一个类中,一个非事务方法调用另一个事务方法,事务不会生效
恭喜你,解决一个 bug 的同时,成功引入了另一个 bug
你懵的同时,你老大也懵
data:image/s3,"s3://crabby-images/a1eaa/a1eaa4f6f05e9885eb5422470804fc89d07ff087" alt=""
你们肯定会问:非事务方法 update 调用事务方法 updateUser ,事务为什么会失效了?
巧了,正好我有答案:记一次线上问题 → 事务去哪了
data:image/s3,"s3://crabby-images/76c54/76c547952a4f202fbf3eee7d3424ccef522b6740" alt=""
别扭拎
同一个类中,非事务方法调用事务方法,事务不生效的解决方案中,是不是有这样一种解决方案:自己注册自己!
data:image/s3,"s3://crabby-images/5b112/5b112b7a42d2622858bd1ba012db018f1057a080" alt=""
我们 debug 一下,看下堆栈情况
我们先看 update
data:image/s3,"s3://crabby-images/b6b8b/b6b8b3bf61b9cc40581fccbb3520726e6559b918" alt=""
调用链中没有事务相关内容
我们再看 updateUser
data:image/s3,"s3://crabby-images/0cf1b/0cf1b007fa9cd579b48892c74442703f664271ae" alt=""
调用链中有事务相关内容
从结果来看,确实能够满足要求,上游调用 update 的地方也不用调整,并且还自给自足,感觉是个好方案呀
但 自己注册自己 这种情况,你们见得多吗,甚至见过吗
反正我看着好别扭,不知道你们有这种感觉没有?
要不将就着这么用?
data:image/s3,"s3://crabby-images/00aea/00aea1d2286ddcc5a66aebb3889a1d201102f54c" alt=""
常规拎
自己注册自己 是非常不推荐的!
为什么不推荐? 来来来,把脸伸过来
data:image/s3,"s3://crabby-images/29760/29760121350531674a15e288f9e71ccde9202782" alt=""
怎么这么多问题,非要把我榨干?
那我就说几点
1、违反了单一职责原则,一个类应该只负责一件事情,如果它开始依赖自己,那么它的职责就不够清晰,这可能会导致代码难以维护和扩展
2、循环依赖,自己依赖自己就是最简单版的循环依赖,虽说 Spring 能解决部分循环依赖,但 Spring 是不推荐循环依赖写法的
3、导致一些莫名其妙的问题,还非常难以排查,大家可以 Google 一下,关键字类似: Spring 自己注入自己 有什么问题
推荐的做法是新建一个 UserManager ,类似如下
data:image/s3,"s3://crabby-images/9765b/9765bb6d3a1901ce2e67b3e5297fff032d4f5ef8" alt=""
此时,上游调用的地方也需要调整,改调用 com.qsl.manager.UserManager#update ,如下所示:
data:image/s3,"s3://crabby-images/e95bf/e95bfe1a99f586448ffb98610fe120fe40be5a17" alt=""
同样 debug 下,来看看堆栈信息
com.qsl.manager.UserManager#update 调用栈情况如下
data:image/s3,"s3://crabby-images/d26c4/d26c419e46c2a5adce26fdcbdf477f8cb0574bf9" alt=""
非常简单,没有任何的代理
我们再看下 com.qsl.service.impl.UserServiceImpl#updateUser
data:image/s3,"s3://crabby-images/d012c/d012cb6cedd4dc5b7430689272f04dd737c5d03f" alt=""
此时,调用链中是有事务相关内容的
是不是很完美的将消息发送从事务中抽出来了?
这确实也是我们最常用的方式,没有之一!
惊喜拎
既不想新增 UserManager ,又想把消息发送从事务中抽离出来,还要保证事务生效,并且不能用 自己注册自己 ,有什么办法吗
data:image/s3,"s3://crabby-images/0bc3e/0bc3eda11397e8227f420b6ef0e78dc7ca306fca" alt=""
好处全都要,坏处往外撂,求求你,做个人吧
data:image/s3,"s3://crabby-images/22697/226971ecce7cce68e25ef9c2fe6e59d00a7cc7bc" alt=""
但是,注意转折来了!
最近我还真学了一个新知识: TransactionSynchronizationManager ,发现它完美契合上述的既要、又要、还要、并且要!
我们先回到最初的版本
data:image/s3,"s3://crabby-images/52f89/52f896d4e33255314890f9210d8027c15ffd2b70" alt=""
接下来看我表演,稍微调整下代码
data:image/s3,"s3://crabby-images/88ebe/88ebe584698990b57844002c6e2c1fed09a1f6a3" alt=""
什么,调整了哪些,看的不够直观?
我真是服了你们这群老六,那我就再爱你们一次,让你们看的更直观,直接 beyond compare 下
data:image/s3,"s3://crabby-images/8cf0c/8cf0c3d59bd8ec804ff4c2138ca2ff806f2312de" alt=""
就调整这么一点,上游调用 update 的地方也不用调整,你们的既要、又要、还要、并且要就满足了!
是不是很简单?
data:image/s3,"s3://crabby-images/ec767/ec7676333cac0bccea74818f90cdffa1d0cc118a" alt=""
为了严谨,我们来验证一下
如何验证了?
最简单的办法就是在发送消息的地方打个断点,如下所示
data:image/s3,"s3://crabby-images/20974/209748250795ef66def75bacdd64793d322fb027" alt=""
当 debug 执行到此的时候,消息是未发送的,这个没问题吧?
那么我们只需要验证:此时事务是否已经提交
问题又来了,如何验证事务已经提交了呢?
很简单,我们直接去数据库查对应的记录,是不是修改之后的数据,如果是,那就说明事务已经提交,否则说明事务没提交,能理解吧?
我们以修改 张三 的密码为例, bebug 未开始,此时 张三 的密码是 zhangsan1
data:image/s3,"s3://crabby-images/657cd/657cd0f790ea61e260be42f9ba27589d646f758c" alt=""
我们把 张三 的密码改成 zhangsan2
data:image/s3,"s3://crabby-images/c249e/c249ef4b3d8c208157b86f4c157b36326c16a6a2" alt=""
开始 bebug
data:image/s3,"s3://crabby-images/b2805/b2805b9702e700e3b1063338699d082777c1d015" alt=""
此时,消息还未发送,我们去数据库查下 张三 的密码
data:image/s3,"s3://crabby-images/b0b70/b0b7062b7225055b466f1bfbf5368db062e71729" alt=""
此时 张三 的密码已经是 zhangsan2 了,是修改之后的数据,说明了什么?
说明事务已经提交了,而此时消息还未发送!
是不是很优雅的实现了最初的重点:把消息发送从事务中拎出来就好了,也就是等事务提交后,再发消息
data:image/s3,"s3://crabby-images/c7358/c7358a83f11c403df15eac0f43ad51bb86dc7391" alt=""
TransactionSynchronizationManager
从字面意思来看,就是一个事务同步管理器
概况
TransactionSynchronizationManager 是 Spring 框架中提供的一个工具类,主要用于管理事务的同步操作
通过 TransactionSynchronizationManager ,开发者可以自定义实现 TransactionSynchronization 接口或继承 TransactionSynchronizationAdapter
data:image/s3,"s3://crabby-images/a6372/a6372718b748dc8080141af75190a78465270708" alt=""
从而在事务的不同阶段(如提交前、提交后、回滚后等)执行特定的操作(如发送消息)
TransactionSynchronizationManager 提供了很多静态方法, registerSynchronization 就是其中之一(其他的大家自行去学习)
data:image/s3,"s3://crabby-images/af425/af4256c7b3e017aa1c17f3e6209489d0b2e0a796" alt=""
入参类型是 TransactionSynchronization ,该接口定义了几个事务同步方法(命名很好,见名知意)
data:image/s3,"s3://crabby-images/d47f0/d47f0712b8818a087d7359849b02577d7f00d161" alt=""
分别代表着在事务的不同阶段,会被执行的操作,比如 afterCommit 会在事务提交后执行
底层原理
为什么事务提交后一定会执行 org.springframework.transaction.support.TransactionSynchronization#afterCommit ?
幕后一定有操盘手,我们来揪一揪它
怎么揪?
正所谓: 源码之下无密码 ,我们直捣黄龙干源码
问题又来了, Spring 源码那么多,我们怎么知道哪一部分跟 TransactionSynchronization 有关?
很简单,去 bebug 的堆栈中找,很容易就能找到切入点
data:image/s3,"s3://crabby-images/e3075/e30755dc6d1e1745a2425faf701a3e271118829c" alt=""
切入点是不是很明显了: org.springframework.transaction.support.AbstractPlatformTransactionManager#commit
/**
* This implementation of commit handles participating in existing
* transactions and programmatic rollback requests.
* Delegates to {@code isRollbackOnly}, {@code doCommit}
* and {@code rollback}.
* @see org.springframework.transaction.TransactionStatus#isRollbackOnly()
* @see #doCommit
* @see #rollback
*/
@Override
public final void commit(TransactionStatus status) throws TransactionException {
if (status.isCompleted()) {
throw new IllegalTransactionStateException(
"Transaction is already completed - do not call commit or rollback more than once per transaction");
}
DefaultTransactionStatus defStatus = (DefaultTransactionStatus) status;
if (defStatus.isLocalRollbackOnly()) {
if (defStatus.isDebug()) {
logger.debug("Transactional code has requested rollback");
}
processRollback(defStatus, false);
return;
}
if (!shouldCommitOnGlobalRollbackOnly() && defStatus.isGlobalRollbackOnly()) {
if (defStatus.isDebug()) {
logger.debug("Global transaction is marked as rollback-only but transactional code requested commit");
}
processRollback(defStatus, true);
return;
}
processCommit(defStatus);
}
View Code
通过 commit 的源码,或者上图的调用链,我们会继续来到 org.springframework.transaction.support.AbstractPlatformTransactionManager#processCommit
/**
* Process an actual commit.
* Rollback-only flags have already been checked and applied.
* @param status object representing the transaction
* @throws TransactionException in case of commit failure
*/
private void processCommit(DefaultTransactionStatus status) throws TransactionException {
try {
boolean beforeCompletionInvoked = false;
try {
boolean unexpectedRollback = false;
prepareForCommit(status);
triggerBeforeCommit(status);
triggerBeforeCompletion(status);
beforeCompletionInvoked = true;
if (status.hasSavepoint()) {
if (status.isDebug()) {
logger.debug("Releasing transaction savepoint");
}
unexpectedRollback = status.isGlobalRollbackOnly();
status.releaseHeldSavepoint();
}
else if (status.isNewTransaction()) {
if (status.isDebug()) {
logger.debug("Initiating transaction commit");
}
unexpectedRollback = status.isGlobalRollbackOnly();
doCommit(status);
}
else if (isFailEarlyOnGlobalRollbackOnly()) {
unexpectedRollback = status.isGlobalRollbackOnly();
}
// Throw UnexpectedRollbackException if we have a global rollback-only
// marker but still didn't get a corresponding exception from commit.
if (unexpectedRollback) {
throw new UnexpectedRollbackException(
"Transaction silently rolled back because it has been marked as rollback-only");
}
}
catch (UnexpectedRollbackException ex) {
// can only be caused by doCommit
triggerAfterCompletion(status, TransactionSynchronization.STATUS_ROLLED_BACK);
throw ex;
}
catch (TransactionException ex) {
// can only be caused by doCommit
if (isRollbackOnCommitFailure()) {
doRollbackOnCommitException(status, ex);
}
else {
triggerAfterCompletion(status, TransactionSynchronization.STATUS_UNKNOWN);
}
throw ex;
}
catch (RuntimeException | Error ex) {
if (!beforeCompletionInvoked) {
triggerBeforeCompletion(status);
}
doRollbackOnCommitException(status, ex);
throw ex;
}
// Trigger afterCommit callbacks, with an exception thrown there
// propagated to callers but the transaction still considered as committed.
try {
triggerAfterCommit(status);
}
finally {
triggerAfterCompletion(status, TransactionSynchronization.STATUS_COMMITTED);
}
}
finally {
cleanupAfterCompletion(status);
}
}
View Code
data:image/s3,"s3://crabby-images/543c8/543c861db731c8b6ea1472188992b0726e4ce34b" alt=""
大家仔细看这个方法,在 doCommit(status) 之前有 triggerBeforeCommit(status) 、 triggerBeforeCompletion(status)
doCommit(status) 之后有 triggerAfterCommit(status) 、 triggerAfterCompletion(status, TransactionSynchronization.STATUS_COMMITTED)
这几个方法的作用很明显了吧( trigger 是触发的意思)
接下来我们跟哪个方法?
很明显,我们要跟 triggerAfterCommit(status) ,因为我们要找的是 afterCommit 的操盘手
data:image/s3,"s3://crabby-images/f15f7/f15f759c51012f2af4a6af39b738d8b2736dfa7b" alt=""
内容很简单,下一步跟的对象也很明确
data:image/s3,"s3://crabby-images/0ce41/0ce417cc5c2c78906ecb1b9f23cb8365e5c519f2" alt=""
这里要分两步说明下
1、 TransactionSynchronizationManager.getSynchronizations()
data:image/s3,"s3://crabby-images/5e65d/5e65d0b39030bd595474bd6137c9c7b52603e2c6" alt=""
先获取所有的事务同步器,然后进行排序
排序先撇开,我们先看看获取到了哪些事务同步器
data:image/s3,"s3://crabby-images/34a03/34a03684ef2c9b1c8a661665ee4ef9e87c0cc9b3" alt=""
第一个不眼熟,我们先不管
第二个眼不眼熟?是不是就是 com.qsl.service.impl.UserServiceImpl#update 中的匿名内部类?(如果想看的更明显,就不要用匿名内部类)
data:image/s3,"s3://crabby-images/ee19e/ee19ec8cf9a4b6bce652fa517a5a0733652d4c42" alt=""
是不是就对应上了:先注册,再获取,最后被调用
被调用就是下面的第 2 步
2、 invokeAfterCommit
data:image/s3,"s3://crabby-images/2bf1f/2bf1f26a1e2dc67f4aa7e1f276cce8f33199d066" alt=""
逻辑很简单,遍历所有事务同步器,逐个调用事务同步器的 afterCommit 方法
我们案例中的 发消息 就是在此处被执行了
至此,相信大家都没疑惑了吧
data:image/s3,"s3://crabby-images/28479/28479f896ce9aaf20459d5f36f1cb44183f9fc99" alt=""
总结
1、关于 Spring 循环依赖,大家可以翻阅下我之前的博客
Spring 的循环依赖,源码详细分析 → 真的非要三级缓存吗
再探循环依赖 → Spring 是如何判定原型循环依赖和构造方法循环依赖的?
四探循环依赖 → 当循环依赖遇上 BeanPostProcessor,爱情可能就产生了!
总之一句话:一定要杜绝循环依赖!
2、事务提交之后再执行某些操作的实现方式
事务失效的方式,大家一定要警惕,这坑很容易掉进去
自己注册自己的方式,直接杜绝,就当没有这种方式
Manager 方式很常规,可以使用
TransactionSynchronizationManager 方式很优雅,推荐使用
看了这篇博客后,该用哪种方式,大家心里有数了吧
3、TransactionSynchronizationManager 使用有限制条件
具体看其注释说明,就当给你们留的家庭作业了
一定要去看,不然使用出了问题可别怪我没提醒你们