Seata源码分析——@GlobalTransactional

Seata源码分析——@GlobalTransactional

  • 前言
  • 脑图
  • Seata三大角色
  • @GlobalTransactional
    • 源码入口
    • GlobalTransactionScanner
      • 初始化TM,RM
      • wrapIfNecessary
  • 分支事务
  • 总结


前言

研读Seata源码有一段时间了,打算出一系列关于源码的文章,旨在加强自己对Seata的理解,同时也希望能帮助到读者。


脑图

本文主要分析标红的地方


Seata三大角色

TC :事务协调者,netty server
TM :事务管理器,netty client
RM: 资源管理器,netty client

  1. 只要方法上加了@GlobalTransactional,Seata通过aop检测到之后,就会使用TM和TC通信,注册全局事务。

  2. 在@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生命周期回调接口

@Overridepublic void afterPropertiesSet() {//是否禁止了全局事务if (disableGlobalTransaction) {if (LOGGER.isInfoEnabled()) {LOGGER.info("Global transaction is disabled.");}return;}//初始化netty 客户端(TM  RM)initClient();}

初始化TM,RM

  private void initClient() {  //init TMTMClient.init(applicationId, txServiceGroup);//init RMRMClient.init(applicationId, txServiceGroup);}

代码最终会调用到RpcClientBootstrap#start

wrapIfNecessary


GlobalTransactionalInterceptor#invoke

GlobalTransactionalInterceptor#handleGlobalTransaction---->TransactionalTemplate#execute

全局事务的处理是通过spring aop来增强的,下面我们来看看分支事务是如何处理的。


分支事务

代理数据源配置
DataSourceProxy#getConnection

@Overridepublic 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作为初学还是挺难的,问题也很多,比如:

  1. 全局锁怎么实现的?
  2. 为什么需要全局锁?
  3. lock_table什么时候插入记录,什么时候删除?
  4. 读写隔离?

这些问题最好先思考在看答案,读者有更好的问题也可以提在评论区,我会更新博客回答,另外seata.io的快速入门关于AT模式的读写隔离真的绝了,把全局锁讲明白了,建议仔细阅读。

问题回答:

  1. 全局锁使用数据库表实现,lock_table。
  2. 全局锁用于读写隔离,如果有多个分布式事务同时操作同一行数据库记录,那么可以保证数据的正确性。
  3. 注册分支事务的时候会插入lock_table记录(正常情况),全局事务提交的时候会删除lock_table。
  4. 写隔离,如果要用分布式事务,那么对于同一张表更新时建议全使用@GlobalTransaction.
    读隔离,使用@GlobalTransactional+select for update 或者 @GlobalLock+@Transactional+select for update

本文链接:https://my.lmcjl.com/post/18515.html

展开阅读全文

4 评论

留下您的评论.