2023-09-16
原文作者:王伟王胖胖 原文地址: https://blog.csdn.net/wangwei19871103/article/details/105366772

创建事务信息流程图

202309162320009741.png

处理提交流程图

202309162320020702.png

处理回滚流程图

202309162320029243.png

开启新连接

上篇说了,会给每个方法创建一个事务状态TransactionStatus,实现类是DefaultTransactionStatus,里面有三个属性很重要,一个是当前新创建的事务transaction,一个是是否是新事务标记newTransaction,另外一个是newSynchronization是否是新同步,也就是新的事务可能需要跟线程做同步,把线程私有变量设置成跟当前事务相关的值。如果设置了新事务,那是什么样的呢,其实就在doBegin里,里面会开启新连接:

202309162320037894.png
如果是第一次创建事务,或者是要创建一个新事务,比如REQUIRES_NEW的情况,是会开一个新的连接的,然后封装成连接持有器ConnectionHolder保存在新的事务里,newConnectionHolder=true

202309162320042915.png
ConnectionHolder有个关键属性newConnectionHolder,能确保这个连接是新创建事务的,还是老的已存在事务的连接。我们在创建新事务的时候,如果有存在老连接的话,会设置老连接进来,这个时候的newConnectionHolderfalse,这个就是能判断这个事务的连接是新创建的,还是已存在事务的。

202309162320051486.png

自动提交

这里会手动把自动提交给关了,后面释放的时候会设置会去的:

202309162320057867.png

标记事务激活了

这个时候标记下这个JDBC连接事务激活了。

202309162320063238.png

数据源和连接资源绑定

这个时候就用到上面的newConnectionHolder啦,如果是新的连接资源的话,就需要让线程私有变量来绑定,因为新创建的事务的连接持有器就是从线程私有变量里取的,这样解决多线程的安全问题。

202309162320067009.png
新创建事务的连接持有器从线程私有变量取:

2023091623200714110.png
绑定到线程私有变量的resources中。

2023091623200774311.png

TransactionSynchronizationManager的线程私有变量

其实线程私有变量有好几个,都是跟事务信息相关的,经常会变,为了解决多线程安全问题,避免用锁,所以用ThreadLocal来存了:

2023091623200894912.png
好了,今天就到这里了,希望对学习理解有帮助,大神看见勿喷,仅为自己的学习理解,能力有限,请多包涵。

阅读全文