前言

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


脑图

本文主要分析标红的地方
Seata源码分析——@GlobalTransactional_全局锁


Seata三大角色

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

Seata源码分析——@GlobalTransactional_GlobalTran_02

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

  2. 在@GlobalTransactional涵括的代码中,不管是本服务中的sql操作,还是feign调用别的服务的sql操作,只要sql操作满足如下:insert操作,delete操作,update操作,select for update操作。 就会被seata增强,使用RM与TC通信,注册分支事务。


@GlobalTransactional

源码入口

Seata源码分析——@GlobalTransactional_GlobalTran_03
GlobalTransactionScanner继承自AbstractAutoProxyCreator,在这里拦截到加了@GlobalTransactional的方法。

GlobalTransactionScanner

Seata源码分析——@GlobalTransactional_seata_04
我们需要关注的方法如下:

  • AbstractAutoProxyCreator:wrapIfNecessary(aop的核心),getAdvicesAndAdvisorsForBean(拦截器)
  • InitializingBean:afterPropertiesSet(初始化TM,RM)

初始化TM,RM

spring生命周期回调接口

@Override
    public void afterPropertiesSet() {
        //是否禁止了全局事务
        if (disableGlobalTransaction) {
            if (LOGGER.isInfoEnabled()) {
                ("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
Seata源码分析——@GlobalTransactional_分布式事务_05

wrapIfNecessary

Seata源码分析——@GlobalTransactional_GlobalTran_06
GlobalTransactionalInterceptor#invoke
Seata源码分析——@GlobalTransactional_seata_07
GlobalTransactionalInterceptor#handleGlobalTransaction---->TransactionalTemplate#execute
Seata源码分析——@GlobalTransactional_GlobalTran_08
全局事务的处理是通过spring aop来增强的,下面我们来看看分支事务是如何处理的。


分支事务

代理数据源配置
Seata源码分析——@GlobalTransactional_全局锁_09DataSourceProxy#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源码分析——@GlobalTransactional_分布式事务_10


总结

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