Seata源码分析——@GlobalTransactional
前言
研读Seata源码有一段时间了,打算出一系列关于源码的文章,旨在加强自己对Seata的理解,同时也希望能帮助到读者。
脑图
本文主要分析标红的地方
Seata三大角色
TC :事务协调者,netty server
TM :事务管理器,netty client
RM: 资源管理器,netty client
-
只要方法上加了@GlobalTransactional,Seata通过aop检测到之后,就会使用TM和TC通信,注册全局事务。
-
在@GlobalTransactional涵括的代码中,不管是本服务中的sql操作,还是feign调用别的服务的sql操作,只要sql操作满足如下:insert操作,delete操作,update操作,select for update操作。 就会被seata增强,使用RM与TC通信,注册分支事务。
@GlobalTransactional
源码入口
GlobalTransactionScanner继承自AbstractAutoProxyCreator,在这里拦截到加了@GlobalTransactional的方法。
GlobalTransactionScanner
我们需要关注的方法如下:
- AbstractAutoProxyCreator:wrapIfNecessary(aop的核心),getAdvicesAndAdvisorsForBean(拦截器)
- InitializingBean:afterPropertiesSet(初始化TM,RM)
初始化TM,RM
spring生命周期回调接口
@Override
public void afterPropertiesSet() {
//是否禁止了全局事务
if (disableGlobalTransaction) {
if (LOGGER.isInfoEnabled()) {
LOGGER.info("Global transaction is disabled.");
}
return;
}
//初始化netty 客户端(TM RM)
initClient();
}
初始化TM,RM
private void initClient() {
//init TM
TMClient.init(applicationId, txServiceGroup);
//init RM
RMClient.init(applicationId, txServiceGroup);
}
代码最终会调用到RpcClientBootstrap#start
wrapIfNecessary
GlobalTransactionalInterceptor#invoke
GlobalTransactionalInterceptor#handleGlobalTransaction---->TransactionalTemplate#execute
全局事务的处理是通过spring aop来增强的,下面我们来看看分支事务是如何处理的。
分支事务
代理数据源配置
DataSourceProxy#getConnection
@Override
public ConnectionProxy getConnection() throws SQLException {
Connection targetConnection = targetDataSource.getConnection();
return new ConnectionProxy(this, targetConnection);
}
ConnectionProxy#doCommit
private void doCommit() throws SQLException {
//处理@GlobalTransaction的分支事务
if (context.inGlobalTransaction()) {
processGlobalTransactionCommit();
}
//处理@GlobalLock,即检查一下是否可以获取全局锁
else if (context.isGlobalLockRequire()) {
processLocalCommitWithGlobalLocks();
} else {
targetConnection.commit();
}
}
ConnectionProxy#processGlobalTransactionCommit
正常情况注册分支事务还会往lock_tabel插入一条记录,代表某个表的某行记录被seata用全局锁锁住了
总结
seata作为初学还是挺难的,问题也很多,比如:
- 全局锁怎么实现的?
- 为什么需要全局锁?
- lock_table什么时候插入记录,什么时候删除?
- 读写隔离?
这些问题最好先思考在看答案,读者有更好的问题也可以提在评论区,我会更新博客回答,另外seata.io的快速入门关于AT模式的读写隔离真的绝了,把全局锁讲明白了,建议仔细阅读。
问题回答:
- 全局锁使用数据库表实现,lock_table。
- 全局锁用于读写隔离,如果有多个分布式事务同时操作同一行数据库记录,那么可以保证数据的正确性。
- 注册分支事务的时候会插入lock_table记录(正常情况),全局事务提交的时候会删除lock_table。
- 写隔离,如果要用分布式事务,那么对于同一张表更新时建议全使用@GlobalTransaction.
读隔离,使用@GlobalTransactional+select for update 或者 @GlobalLock+@Transactional+select for update