目录
1、mybatis事物管理器
2、SpringManagedTransactionFactory如何处理事物
3、spring事物如何设置connection连接到threadLocal
1、mybatis事物管理器
mybatis事物抽象接口类:Transaction。该接口定义了事物基本方法和获取数据库连接方法
该类有三个实现类JdbcTransaction、ManagedTransaction、SpringManagedTransaction。
其中JdbcTransaction、ManagedTransaction是mybatis自己使用,不是我们关注的对象。我们重点关注SpringManagedTransaction,在spring环境中,mybatis使用该类来处理事物。
既然明确了是使用该类来处理事物,那我们先来看看在哪里初始化SpringManagedTransaction。老套路,ctrl+f, 发现只有在 SpringManagedTransactionFactory类中调用
继续查看 SpringManagedTransactionFactory 在哪里调用,发现是在org.mybatis.spring.SqlSessionFactoryBean#buildSqlSessionFactory()方法中初始化
mybatis-spring中,是通过SqlSessionFactoryBean#buildSqlSessionFactory()方法创建
SqlSessionFactory, 即说明mybatis-spring中默认使用SpringManagedTransactionFactory 为事物管理器。
2、SpringManagedTransactionFactory如何处理事物
从SpringManagedTransaction#getConnection()方法开始,到重点方法
DataSourceUtils.getConnection(this.dataSource)。其中DataSourceUtils类是spring提供的工具类
跟进去,查看DataSourceUtils#doGetConnection()源码,发现先从spring提供的threadLocal中获取connection连接,如果没有则重新获取,并且会通过 TransactionSynchronizationManager.isSynchronizationActive() 判断当前是否处于spring事物中,如果是则需要将connection连接绑定到spring threadLocal中。
至此,已经解决了mybatis是如何获取spring事物中的连接的。既然有获取,就有地方设置,那么下面我们再看spring事物是如何设置connection连接到threadLocal中的。
3、spring事物如何设置connection连接到threadLocal
熟悉spring事物的同学都知道,spring事物的增强是在TransactionInterceptor中的,那么我们直接看该方法的invoke方法
继续看核心方法TransactionAspectSupport#invokeWithinTransaction()方法
我们重点看TransactionAspectSupport#createTransactionIfNecessary()方法是如何处理的,跟进去最后会到TransactionAspectSupport#prepareTransactionInfo()方法,
最后会到TransactionAspectSupport.TransactionInfo#bindToThread()方法,继续跟进去
那么我们再看,发现和 DataSourceUtils 没有丝毫关系。说明我们找错了地方。
再次查看 TransactionSynchronizationManager#bindResource()方法调用地方,发现是在
AbstractPlatformTransactionManager#startTransaction()方法中绑定。那么
需要查看 AbstractPlatformTransactionManager 和 TransactionInterceptor 的关系。
发现 在初始化 TransactionInterceptor的时候,会设置 DataSourceTransactionManager。
再回头查看,发现此处会调用AbstractPlatformTransactionManager#getTransaction()方法。
继续跟,发现 此处会调用子类 DataSourceTransactionManager#doGetTransaction()方法
由于 doGetTransaction() 方法为获取已存在的事物,所以我们重点看新建事物的处理
在 DataSourceTransactionManager#doBegin()方法中,可以看到将新建的事物绑定到当前线程上下文