美文网首页
seata TCC模式

seata TCC模式

作者: 紫色红色黑色 | 来源:发表于2019-10-29 23:59 被阅读0次

描述

TCC是相对于AT比较偏上层。是针对接口而实现的。要求全局事务发起者接口添加@GlobalTransactional,参与者接口prepare()添加@TwoPhaseBusinessAction,并制定commit方法和rollback方法。

适合在暴露RPC接口时使用,在OutService层方法中添加@GlobalTransactional开启全局事务。在InnerService层中方法中添加@TwoPhaseBusinessAction(name = "", commitMethod = "commit", rollbackMethod = "rollback")注解指定commit和rollback方法

spring中配置GlobalTransactionScanner初始化。该类实现AbstractAutoProxyCreatorInitializingBean。首先会执行wrapIfNecessary()

初始化

1.注册资源,封装TCCResource并缓存。生成ResourceId,将ResourceId注册到TC
2.设置@TwoPhaseBusinessAction的拦截器为TccActionInterceptor,设置@GlobalTransactional的拦截器为GlobalTransactionalInterceptor

一阶段

TM

TM处理流程和AT模式下一样,
1.开启全局事务,向TC注册全局事务并返回XID
2.如果业务执行成功,通知TC全局事务提交
3.如果业务执行失败,通知TC全局事务回滚
4.清除内存中XID

RM

注册分支事务,获取branchId。并将方法调用时的上下文发送给TC。执行业务逻辑

public Map<String, Object> proceed(Method method, Object[] arguments, String xid, TwoPhaseBusinessAction businessAction,
                                    Callback<Object> targetCallback) throws Throwable {
    Map<String, Object> ret = new HashMap<String, Object>(16);

    //TCC name
    String actionName = businessAction.name();
    BusinessActionContext actionContext = new BusinessActionContext();
    actionContext.setXid(xid);
    //set action anme
    actionContext.setActionName(actionName);
    //TODO services

    //Creating Branch Record
    String branchId = doTccActionLogStore(method, arguments, businessAction, actionContext);
    actionContext.setBranchId(branchId);

    //set the parameter whose type is BusinessActionContext
    Class<?>[] types = method.getParameterTypes();
    int argIndex = 0;
    for (Class<?> cls : types) {
        if (cls.getName().equals(BusinessActionContext.class.getName())) {
            arguments[argIndex] = actionContext;
            break;
        }
        argIndex++;
    }
    //the final parameters of the try method
    ret.put(Constants.TCC_METHOD_ARGUMENTS, arguments);
    //the final result
    ret.put(Constants.TCC_METHOD_RESULT, targetCallback.execute());
    return ret;
}

二阶段

TM通知TC全局提交/回滚。TC通知各分支事务。RM处理消息逻辑是RMHandlerTCC里。

RM

提交和回滚逻辑一样。
1.用TC发过来的ResourceId查到TCResource,找到commit/rollback方法并执行
2.通知TC执行结果

public BranchStatus branchCommit(BranchType branchType, String xid, long branchId, String resourceId,
                                    String applicationData) throws TransactionException {
    TCCResource tccResource = (TCCResource)tccResourceCache.get(resourceId);
    if (tccResource == null) {
        throw new ShouldNeverHappenException("TCC resource is not exist, resourceId:" + resourceId);
    }
    Object targetTCCBean = tccResource.getTargetBean();
    Method commitMethod = tccResource.getCommitMethod();
    if (targetTCCBean == null || commitMethod == null) {
        throw new ShouldNeverHappenException("TCC resource is not available, resourceId:" + resourceId);
    }
    try {
        boolean result = false;
        //BusinessActionContext
        BusinessActionContext businessActionContext = getBusinessActionContext(xid, branchId, resourceId,
            applicationData);
        Object ret = commitMethod.invoke(targetTCCBean, businessActionContext);
        LOGGER.info(
            "TCC resource commit result :" + ret + ", xid:" + xid + ", branchId:" + branchId + ", resourceId:"
                + resourceId);
        if (ret != null) {
            if (ret instanceof TwoPhaseResult) {
                result = ((TwoPhaseResult)ret).isSuccess();
            } else {
                result = (boolean)ret;
            }
        }
        return result ? BranchStatus.PhaseTwo_Committed : BranchStatus.PhaseTwo_CommitFailed_Retryable;
    } catch (Throwable t) {
        String msg = String.format("commit TCC resource error, resourceId: %s, xid: %s.", resourceId, xid);
        LOGGER.error(msg, t);
        throw new FrameworkException(t, msg);
    }
}

引用

https://zhuanlan.zhihu.com/p/61981170
https://blog.csdn.net/hosaos/article/details/89847554

相关文章

  • seata-golang 接入指南

    seata-golang 是一个分布式事务框架,实现了 AT 模式和 TCC 模式,AT 模式相较 TCC 模式对...

  • Seata in AT mode的工作原理

    Seata为用户提供了AT、TCC、SAGA和XA事务模式。其中AT模式是seata主推的事务模式,使用AT有一个...

  • seata TCC模式

    描述 TCC是相对于AT比较偏上层。是针对接口而实现的。要求全局事务发起者接口添加@GlobalTransacti...

  • 分布式事务 Seata AT模式原理与实战

    Seata 是阿里开源的基于Java的分布式事务解决方案 AT,XA,TCC,Saga Seata 提供四种模式解...

  • XA 事务水很深,小伙子我怕你把握不住!

    @[toc]分布式事务系列继续! 前面松哥和大家聊了 Seata 中的 TCC 模式以及 AT 模式,没看的小伙伴...

  • Seate

    Seata是阿里开源的一个分布式事务框架。Seata主要有两种分布式事务实现方案,AT及TCC AT模式主要关注多...

  • AT模式遇到的问题与具体源码篇

    关于AT模式的流程,我们在上一篇文章 一文简要概述Seata AT与TCC的区别[https://www.jian...

  • 数据库

    分布式事务框架https://github.com/seata/seata 国内主要的开源TCC分布式事务框架包括...

  • 16. 分布式事务seata的四种模式

    Seata四种分布式事务解决方案: XA模式:强一致性分阶段事务模式,牺牲了一定的可用性,无业务侵入 TCC模式:...

  • Seata- TCC

    1 dubbo接口用twophasebuisseness标注一组tcc接口时,调用方和被调用方都会有接口,如何识别...

网友评论

      本文标题:seata TCC模式

      本文链接:https://www.haomeiwen.com/subject/npmgvctx.html