分布式事务Seata——AT模式解析

AT 模式下,把每个数据库被当做是一个 Resource,Seata 里称为 DataSource Resource。业务通过 JDBC 标准接口访问数据库资源时,Seata 框架会对所有请求进行拦截,做一些操作。每个本地事务提交时,Seata RM(Resource Manager,资源管理器) 都会向 TC(Transaction Coordinator,事务协调器) 注册一个分支事务。当请求链路调用完成后,发起方通知 TC 提交或回滚分布式事务,进入二阶段调用流程。此时,TC 会根据之前注册的分支事务回调到对应参与者去执行对应资源的第二阶段。TC 是怎么找到分支事务与资源的对应关系呢?每个资源都有一个全局唯一的资源 ID,并且在初始化时用该 ID 向 TC 注册资源。在运行时,每个分支事务的注册都会带上其资源 ID。这样 TC 就能在二阶段调用时正确找到对应的资源。

Seata 中有三大模块,分别是 TM、RM 和 TC。 其中 TM 和 RM 是作为 Seata 的客户端与业务系统集成在一起,TC 作为 Seata 的服务端独立部署。


Seata

在 Seata 中,分布式事务的执行流程:

  • TM 开启分布式事务(TM 向 TC 注册全局事务记录);
  • 按业务场景,编排数据库、服务等事务内资源(RM 向 TC 汇报资源准备状态 );
  • TM 结束分布式事务,事务一阶段结束(TM 通知 TC 提交/回滚分布式事务);
  • TC 汇总事务信息,决定分布式事务是提交还是回滚;
  • TC 通知所有 RM 提交/回滚 资源,事务二阶段结束。

Seata 会有 4 种分布式事务解决方案,分别是 AT 模式、TCC 模式、Saga 模式和 XA 模式,本文主要介绍AT模式。

Seata类图

TC类图

TC在Seata中其实就是Server端,Client端其实属于RM和TM,但是为了看起来完整,把Client端的RpcCient和Listener也放在了这里,类图如下图所示:


TC 类图

CLient listenner监听服务端发过来的消息,从而进行相应的操作。

RM类图

RM相关类图如下图所示:


RM 类图

TM类图

TM相关类图如下图所示:


TM 类图

AT模式

AT 模式是一种无侵入的分布式事务解决方案。在 AT 模式下,用户只需关注自己的“业务 SQL”,用户的 “业务 SQL” 作为一阶段,Seata 框架会自动生成事务的二阶段提交和回滚操作。


AT 模式

在分析AT模式前,我们先提问几个问题:

1、分布式Seata,怎么在系统中集成使用?
2、开始事务前,做了什么初始化工作?
3、AT模式具体怎么实现的?

初始化工作

Seata在GlobalTransactionScanner中进行了TM和RM初始化:

private void initClient() {
    //init TM
    TMClient.init(applicationId, txServiceGroup);

    //init RM
    RMClient.init(applicationId, txServiceGroup);
    
    //Register Spring ShutdownHook
    registerSpringShutdownHook();
}

其中TMClient的初始化如下:

public static void init(String applicationId, String transactionServiceGroup) {
    TmRpcClient tmRpcClient = TmRpcClient.getInstance(applicationId, transactionServiceGroup);
    tmRpcClient.init();
}

其中初始化了一个TmRpcClient对象,它的init方法中,会启动与Server的重连接线程(每5秒),以及消息发送线程和超时线程。重连接事,按照配置方式,寻找Server信息,比如采用file形式,那么首先从file.conf中根据分组名称(service_group)找到集群名称(cluster_name),再根据集群名称找到fescar-server集群ip端口列表,然后从ip列表中选择一个用netty进行连接。

RmClient的初始化如下:

public static void init(String applicationId, String transactionServiceGroup) {
    RmRpcClient rmRpcClient = RmRpcClient.getInstance(applicationId, transactionServiceGroup);
    rmRpcClient.setResourceManager(DefaultResourceManager.get());
    rmRpcClient.setClientMessageListener(new RmMessageListener(DefaultRMHandler.get()));
    rmRpcClient.init();
}

其中,主要工作为:

  • 创建一个RmRpcClient对象
  • RmRpcClient对象设置ResourceManager;
  • RmRpcClient对象设置消息Listener;
  • RmRpcClient对象初始化

初始化阶段总结:
1、Spring启动时,初始化了2个客户端TmClient、RmClient
2、TmClient与Server通过Netty建立连接并发送消息
3、RmClient与Server通过Netty建立连接,负责接收二阶段提交、回滚消息并在回调器(RmHandler)中做处理

因此,使用Seata,需要初始化GlobalTransactionScanner进行初始化工作!

TM全局事务控制

那些方法需要进行全局事务控制了?Seata定义了注解GlobalTransactional,只要添加了注解的方法,就代表需要分布式事务。

注解对应着拦截器,Seata定义的拦截器为GlobalTransactionalInterceptor。其核心处理逻辑如下:

@Override
public Object invoke(final MethodInvocation methodInvocation) throws Throwable {
    Class<?> targetClass = (methodInvocation.getThis() != null ? AopUtils.getTargetClass(methodInvocation.getThis()) : null);
    Method specificMethod = ClassUtils.getMostSpecificMethod(methodInvocation.getMethod(), targetClass);
    final Method method = BridgeMethodResolver.findBridgedMethod(specificMethod);

    final GlobalTransactional globalTransactionalAnnotation = getAnnotation(method, GlobalTransactional.class);
    final GlobalLock globalLockAnnotation = getAnnotation(method, GlobalLock.class);
    if (globalTransactionalAnnotation != null) {
        return handleGlobalTransaction(methodInvocation, globalTransactionalAnnotation);
    } else if (globalLockAnnotation != null) {
        return handleGlobalLock(methodInvocation);
    } else {
        return methodInvocation.proceed();
    }
}

上面方法的逻辑为:如果业务方法上有全局事务GlobalTransactional注解时,就执行全局事务处理handleGlobalTransaction;如果业务方法上有全局锁GlobalLock注解时,就执行全局锁处理handleGlobalLock;否则,就按照普通方法执行。

handleGlobalTransaction中,会调用TransactionalTemplate的execute方法,execute方法如下:

public Object execute(TransactionalExecutor business) throws Throwable {
    // 1. get or create a transaction
    GlobalTransaction tx = GlobalTransactionContext.getCurrentOrCreate();

    // 1.1 get transactionInfo
    TransactionInfo txInfo = business.getTransactionInfo();
    if (txInfo == null) {
        throw new ShouldNeverHappenException("transactionInfo does not exist");
    }
    try {
        // 2. begin transaction
        beginTransaction(txInfo, tx);
        
        Object rs = null;
        try {
            // 3. Do Your Business
            rs = business.execute();
        } catch (Throwable ex) {
            // 4.the needed business exception to rollback.
            completeTransactionAfterThrowing(txInfo,tx,ex);
            throw ex;
        }

        // 5. everything is fine, commit.
        commitTransaction(tx);

        return rs;
    } finally {
        //6. clear
        triggerAfterCompletion();
        cleanUp();
    }
}

1、创建或创建一个GlobalTransaction,并根据业务TransactionalExecutor获取到事务需要的相关信息;

2、开始全局事务

开始全局事务的方法为beginTransaction方法,方法内部调用了DefaultGlobalTransaction.begin方法,而DefaultGlobalTransaction.begin方法的内部又调用了TransactionManager的begin方法,

xid = transactionManager.begin(null, null, name, timeout);

TransactionManager接口类的默认实现为DefaultTransactionManager,其begin方法为:

public String begin(String applicationId, String transactionServiceGroup, String name, int timeout)
    throws TransactionException {
    GlobalBeginRequest request = new GlobalBeginRequest();
    request.setTransactionName(name);
    request.setTimeout(timeout);
    GlobalBeginResponse response = (GlobalBeginResponse)syncCall(request);
    return response.getXid();
}

方法中,向服务器发起开始全局事务请求,服务会返回一个全局事务标示XID。OK,此时,由TM发起的一个全局事务就开始了!

3、执行业务逻辑

4、回滚全局事务

如果业务都执行没有全部成功,那么就执行全局事务回滚,方法为completeTransactionAfterThrowing。方法执行逻辑与开始事务方法内部逻辑基本一致:内部调用了DefaultGlobalTransaction.rollback方法,而DefaultGlobalTransaction.rollback方法的内部又调用了TransactionManager的rollback方法,向服务器发起rollback,服务会返回GlobalStatus.

5、提交全局事务

如果业务都执行成功,那么就全局commit,方法为commitTransaction。方法执行逻辑与开始事务方法和回滚事务内部逻辑基本一致:内部调用了DefaultGlobalTransaction.commit方法,而DefaultGlobalTransaction.commit方法的内部又调用了TransactionManager的commit方法,向服务器发起Commit,服务会返回GlobalStatus.

6、清理
完成整个事务后,进行相关资源变量清理。

AT模式一阶段

在一阶段,Seata 会拦截“业务 SQL”,首先解析 SQL 语义,找到“业务 SQL”要更新的业务数据,在业务数据被更新前,将其保存成“before image”,然后执行“业务 SQL”更新业务数据,在业务数据更新之后,再将其保存成“after image”,最后生成行锁。以上操作全部在一个数据库事务内完成,这样保证了一阶段操作的原子性。


AT 模式阶段1

实现上,Seata对数据源做了封装代理,然后对于数据源的操作处理,就由Seata内部逻辑完成了。看一个Demo中的数据源加载配置:

@Bean(name = "order")
public DataSourceProxy masterDataSourceProxy(@Qualifier("originOrder") DataSource dataSource) {
    return new DataSourceProxy(dataSource);
}

从Demo中可以看到,我们使用的是Seata封装的代理数据源DataSourceProxy。DataSourceProxy初始化时,会进行Resouce注册:

private void init(DataSource dataSource, String resourceGroupId) {
    this.resourceGroupId = resourceGroupId;
    Connection connection = dataSource.getConnection()
    jdbcUrl = connection.getMetaData().getURL();
    dbType = JdbcUtils.getDbType(jdbcUrl, null);
    DefaultResourceManager.get().registerResource(this);
}

Seata除了对数据库的DataSource进行了封装,同样也对Connection,Statement进行了封装代理,分别为ConnectionProxy和StatementProxy。

DataSourceProxy中重写了DataSource的getConnection方法,以此来获得ConnectionProxy,方法如下:

@Override
public ConnectionProxy getConnection() throws SQLException {
    Connection targetConnection = targetDataSource.getConnection();
    return new ConnectionProxy(this, targetConnection);
}

ConnectionProxy代理了Sql connection。我们重点关于一下其createStatement方法,prepareStatement方法,commit方法,rollback方法以及setAutoCommit方法。

ConnectionProxy的createStatement会返回一个代理的StatementProxy,如下:

@Override
public Statement createStatement() throws SQLException {
    Statement targetStatement = getTargetConnection().createStatement();
    return new StatementProxy(this, targetStatement);
}

ConnectionProxy的prepareStatement同样会返回一个代理的PreparedStatementProxy,如下:

@Override
public PreparedStatement prepareStatement(String sql) throws SQLException {
    PreparedStatement targetPreparedStatement = getTargetConnection().prepareStatement(sql);
    return new PreparedStatementProxy(this, targetPreparedStatement, sql);
}

StatementProxy和PreparedStatementProxy是对执行Sql的Statement的一个封装,两者基本一样,我们看看StatementProxy的execute方法:

@Override
public boolean execute(String sql) throws SQLException {
    this.targetSQL = sql;
    return ExecuteTemplate.execute(this, new StatementCallback<Boolean, T>() {
        @Override
        public Boolean execute(T statement, Object... args) throws SQLException {
            return statement.execute((String) args[0]);
        }
    }, sql);
}

可以看到,内部交给了ExecuteTemplate执行,ExecuteTemplate的execute方法如下:

public static <T, S extends Statement> T execute(SQLRecognizer sqlRecognizer,
                                             StatementProxy<S> statementProxy,
                                                 StatementCallback<T, S> statementCallback,
                                                 Object... args) throws SQLException {

    if (!RootContext.inGlobalTransaction() && !RootContext.requireGlobalLock()) {
        // Just work as original statement
        return statementCallback.execute(statementProxy.getTargetStatement(), args);
    }

    if (sqlRecognizer == null) {
        sqlRecognizer = SQLVisitorFactory.get(
                statementProxy.getTargetSQL(),
                statementProxy.getConnectionProxy().getDbType());
    }
    Executor<T> executor = null;
    if (sqlRecognizer == null) {
        executor = new PlainExecutor<T, S>(statementProxy, statementCallback);
    } else {
        switch (sqlRecognizer.getSQLType()) {
            case INSERT:
                executor = new InsertExecutor<T, S>(statementProxy, statementCallback, sqlRecognizer);
                break;
            case UPDATE:
                executor = new UpdateExecutor<T, S>(statementProxy, statementCallback, sqlRecognizer);
                break;
            case DELETE:
                executor = new DeleteExecutor<T, S>(statementProxy, statementCallback, sqlRecognizer);
                break;
            case SELECT_FOR_UPDATE:
                executor = new SelectForUpdateExecutor<T, S>(statementProxy, statementCallback, sqlRecognizer);
                break;
            default:
                executor = new PlainExecutor<T, S>(statementProxy, statementCallback);
                break;
        }
    }
    T rs = null;
    try {
        rs = executor.execute(args);
    } catch (Throwable ex) {
        if (!(ex instanceof SQLException)) {
            // Turn other exception into SQLException
            ex = new SQLException(ex);
        }
        throw (SQLException)ex;
    }
    return rs;
}

方法内部流程如下:
1)先判断是否开启了全局事务,如果没有,不走代理;
2)调用SQLVisitorFactory对目标sql进行解析
3)针对特定类型sql操作(INSERT,UPDATE,DELETE,SELECT_FOR_UPDATE)等进行特殊解析
4)执行sql并返回结果

InsertExecutor,UpdateExecutor,DeleteExecutor均继承自AbstractDMLBaseExecutor,AbstractDMLBaseExecutor又继承自BaseTransactionalExecutor,BaseTransactionalExecutor又继承自Executor。其中execute方法实现为:

@Override
public Object execute(Object... args) throws Throwable {
    if (RootContext.inGlobalTransaction()) {
        String xid = RootContext.getXID();
        statementProxy.getConnectionProxy().bind(xid);
    }

    if (RootContext.requireGlobalLock()) {
        statementProxy.getConnectionProxy().setGlobalLockRequire(true);
    } else {
        statementProxy.getConnectionProxy().setGlobalLockRequire(false);
    }
    return doExecute(args);
}

其中doExecute方法如下:

@Override
public T doExecute(Object... args) throws Throwable {
    AbstractConnectionProxy connectionProxy = statementProxy.getConnectionProxy();
    if (connectionProxy.getAutoCommit()) {
        return executeAutoCommitTrue(args);
    } else {
        return executeAutoCommitFalse(args);
    }
}

其中executeAutoCommitFalse方法为:

protected T executeAutoCommitFalse(Object[] args) throws Throwable {
    TableRecords beforeImage = beforeImage();
    T result = statementCallback.execute(statementProxy.getTargetStatement(), args);
    TableRecords afterImage = afterImage(beforeImage);
    prepareUndoLog(beforeImage, afterImage);
    return result;
}

其中首先获取执行Sql前的beforeImage,然后执行Sql,执行晚,获取执行后的afterImage,并将执行前后Image赋给ConnectionProxy。注意插入,删除,更新SQl语句,获取beforeImage和afterImage的方法不同,因此定义了InsertExecutor,UpdateExecutor,DeleteExecutor。

执行完Sql,此时需要提交,ConnectionProxy的commit方法如下:

@Override
public void commit() throws SQLException {
    if (context.inGlobalTransaction()) {
        processGlobalTransactionCommit();
    } else if (context.isGlobalLockRequire()) {
        processLocalCommitWithGlobalLocks();
    } else {
        targetConnection.commit();
    }
}

方法逻辑为:
1)如果处于全局事务中,则调用processGlobalTransactionCommit()处理全局事务提交
2)如果加了全局锁注解,加全局锁并提交
3)如果没有对应注释,按直接进行事务提交

重点关于一下processGlobalTransactionCommit代码:

private void processGlobalTransactionCommit() throws SQLException {
    try {
        register();
    } catch (TransactionException e) {
        recognizeLockKeyConflictException(e);
    }

    try {
        if (context.hasUndoLog()) {
            UndoLogManager.flushUndoLogs(this);
        }
        targetConnection.commit();
    } catch (Throwable ex) {
        report(false);
        if (ex instanceof SQLException) {
            throw new SQLException(ex);
        }
    }
    report(true);
    context.reset();
}

流程分为如下几步:

1、注册分支事务register(),并将branchId分支id绑定到上下文中:

private void register() throws TransactionException {
    Long branchId = DefaultResourceManager.get().branchRegister(BranchType.AT, getDataSourceProxy().getResourceId(),
            null, context.getXid(), null, context.buildLockKeys());
    context.setBranchId(branchId);
}

2、如果包含undolog,则将之前绑定到上下文中的undolog进行入库:

UndoLogManager.flushUndoLogs(this) ;

前面提到,SQl执行时,会生成beforeImage和afterImage,最为一个Unlog赋给ConneectionProxy,此时这里就是将这些Unlog保存到数据库,SQl语句为:

private static String INSERT_UNDO_LOG_SQL = "INSERT INTO " + UNDO_LOG_TABLE_NAME +
    " (branch_id, xid, context, rollback_info, log_status, log_created, log_modified)" +
    " VALUES (?, ?, ?, ?, ?, now(), now())";

3、提交本地事务;

4、如果操作失败,report()中通过RM提交Branch失败消息,如果成功,report()提交Branch成功消息,其中Report方法如下:

private void report(boolean commitDone) throws SQLException {
    int retry = REPORT_RETRY_COUNT;
    while (retry > 0) {
        try {
          DefaultResourceManager.get().branchReport(BranchType.AT, context.getXid(), context.getBranchId(),
                    (commitDone ? BranchStatus.PhaseOne_Done : BranchStatus.PhaseOne_Failed), null);
            return;
        } catch (Throwable ex) {
            retry--;
            if (retry == 0) {
                throw new SQLException("Failed to report branch status " + commitDone, ex);
            }
        }
    }
}

AT模式二阶段

AT时分为两个阶段的,第一阶段,就是各个阶段本地提交操作;第二阶段会根据第一阶段的情况决定是进行全局提交还是全局回滚操作。

全局提交回滚操作由TM发起,具体为,如果Branch执行没有出现异常,那么就表明各个Branch均执行成功,即进行全局提交,如果某个Branch执行时出现异常,那么就需要进行全局回滚。代码即为TransactionalTemplate的execute的第4步和5步,如下:

try {
    // 3. Do Your Business
    rs = business.execute();
} catch (Throwable ex) {
    // 4.the needed business exception to rollback.
    completeTransactionAfterThrowing(txInfo,tx,ex);
    throw ex;
}

// 5. everything is fine, commit.
commitTransaction(tx);

return rs;

Seata Serer 收到TM发送的提交或者回滚请求后,就会向各个RM 发送消息,让他们执行相应的提交或者回滚操作。 RM接收消息是通过RMHandlerAT,在初始化RmListrenner时,指定了RMHandlerAT,其中处理方法为:

@Override
public BranchCommitResponse handle(BranchCommitRequest request) {
    BranchCommitResponse response = new BranchCommitResponse();
    exceptionHandleTemplate(new AbstractCallback<BranchCommitRequest, BranchCommitResponse>() {
        @Override
        public void execute(BranchCommitRequest request, BranchCommitResponse response)
            throws TransactionException {
            doBranchCommit(request, response);
        }
    }, request, response);
    return response;
}

@Override
public BranchRollbackResponse handle(BranchRollbackRequest request) {
    BranchRollbackResponse response = new BranchRollbackResponse();
    exceptionHandleTemplate(new AbstractCallback<BranchRollbackRequest, BranchRollbackResponse>() {
        @Override
        public void execute(BranchRollbackRequest request, BranchRollbackResponse response)
            throws TransactionException {
            doBranchRollback(request, response);
        }
    }, request, response);
    return response;
}

二阶段提交

二阶段如果是提交的话,因为“业务 SQL”在一阶段已经提交至数据库, 所以 Seata 框架只需将一阶段保存的快照数据和行锁删掉,完成数据清理即可。

阶段2提交

如果所有Branch RM都执行成功了,那么就进行全局Commit。因为此时我们不用回滚,而每个Branch本地数据库操作已经完成了,那么我们其实主要做的事情就是把本地的Undolog删了即可,看看Seata内部实现逻辑。

doBranchCommit方法内部会调用ResourceManager的branchCommit方法:

BranchStatus status = getResourceManager().branchCommit(request.getBranchType(), xid, branchId, resourceId,applicationData);

根据Seata SPI,其实就是执行的DataSourceManager类的branchCommit,如下:

@Override
public BranchStatus branchCommit(BranchType branchType, String xid, long branchId, String resourceId, String applicationData) throws TransactionException {
    return asyncWorker.branchCommit(branchType, xid, branchId, resourceId, applicationData);
}

DataSourceManager中调用了asyncWorker来异步提交,看下AsyncWorker中branchCommit方法:

@Override
public BranchStatus branchCommit(BranchType branchType, String xid, long branchId, String resourceId,
                                 String applicationData) throws TransactionException {
    if (!ASYNC_COMMIT_BUFFER.offer(new Phase2Context(branchType, xid, branchId, resourceId, applicationData))) {
        LOGGER.warn("Async commit buffer is FULL. Rejected branch [" + branchId + "/" + xid
            + "] will be handled by housekeeping later.");
    }
    return BranchStatus.PhaseTwo_Committed;
}

方法中往ASYNC_COMMIT_BUFFER缓冲List中新增了一个二阶段提交的context,那么真正branchCommit在哪里了?

其实是在AsyncWorker的init()方法中,Init中会起一个定时线程,每一秒执行一次,线程中的主要逻辑为:
1)先按resourceId(也就是数据连接)对提交操作进行分组,一个数据库的可以一起操作,提升效率;
2) 根据resourceId找到对应DataSourceProxy,并获取一个普通的数据库连接getPlainConnection(),估计这本身不需要做代理操作,故用了普通的数据库连接;
3)调用UndoLogManager.deleteUndoLog(commitContext.xid, commitContext.branchId, conn)删除undolog

二阶段回滚

二阶段如果是回滚的话,Seata 就需要回滚一阶段已经执行的“业务 SQL”,还原业务数据。回滚方式便是用“before image”还原业务数据;但在还原前要首先要校验脏写,对比“数据库当前业务数据”和 “after image”,如果两份数据完全一致就说明没有脏写,可以还原业务数据,如果不一致就说明有脏写,出现脏写就需要转人工处理。

阶段2回滚

AT 模式的一阶段、二阶段提交和回滚均由 Seata 框架自动生成,用户只需编写“业务 SQL”,便能轻松接入分布式事务,AT 模式是一种对业务无任何侵入的分布式事务解决方案。

如果某个RM Branch执行失败了,那么就要进行全局回滚。RMHandlerAT收到回滚消息,进行回滚处理,内部会调用DataSourceManager的branchRollback方法:

@Override
public BranchStatus branchRollback(BranchType branchType, String xid, long branchId, String resourceId, String applicationData) throws TransactionException {
    DataSourceProxy dataSourceProxy = get(resourceId);
    if (dataSourceProxy == null) {
        throw new ShouldNeverHappenException();
    }
    try {
        UndoLogManager.undo(dataSourceProxy, xid, branchId);
    } catch (TransactionException te) {
        if (te.getCode() == TransactionExceptionCode.BranchRollbackFailed_Unretriable) {
            return BranchStatus.PhaseTwo_RollbackFailed_Unretryable;
        } else {
            return BranchStatus.PhaseTwo_RollbackFailed_Retryable;
        }
    }
    return BranchStatus.PhaseTwo_Rollbacked;

}

交由UndoLogManager执行undo操作,undo中主要分为以下几步:
1、首先去数据库查询,是否有undolog,sql语句为:

private static String SELECT_UNDO_LOG_SQL = "SELECT * FROM " + UNDO_LOG_TABLE_NAME WHERE branch_id = ? AND xid = ? FOR UPDATE";

2、数据库中有如有undolog,那么查出来构造BranchUndoLog对象,一条log对应一个对象;
3、对于每一个BranchUndoLog对象,内部可能包含多个SQLUndoLog,遍历BranchUndoLog中的SQLUndoLog List,分别进行处理;

for (SQLUndoLog sqlUndoLog : branchUndoLog.getSqlUndoLogs()) {
    TableMeta tableMeta = TableMetaCache.getTableMeta(dataSourceProxy, sqlUndoLog.getTableName());
    sqlUndoLog.setTableMeta(tableMeta);
    AbstractUndoExecutor undoExecutor = UndoExecutorFactory.getUndoExecutor(
            dataSourceProxy.getDbType(),
            sqlUndoLog);
    undoExecutor.executeOn(conn);
}

处理过程就是,根实际操作反着来,比如我们插入一条数据,那么就删除这条数据;如果修改了一条数据,那么就给他修改回去。相应的undoExecutor分别为:MySQLUndoDeleteExecutor,MySQLUndoInsertExecutor,MySQLUndoUpdateExecutor。因为我们第一阶段执行本地操作时,保存了beforeImage和afterImage,所以反着构建SQl时,可以拿到原来的数据。

后记

看完本文,你是否可以回答文章前面提到的几个问题了?如果不可以,那就再亲自去看看代码吧,最好的文档就是代码。

祝 工作顺利,天天开心!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352

推荐阅读更多精彩内容