Seata中的TCC模式学习一

简介: Seata中的TCC模式学习一

由于我们的客户中心发送线索、整车订单服务配车、整车财务服务财务信息推送中都使用的全局事务是基于@GlobalTransactional、@TwoPhaseBusinessAction,因此有必要了解这两个注解中是如何在分布式事务中发挥作用的,也即TCC的实现。虽然AT模式比TCC模式要好。

一、TCC实现模式的分类

资源预留模式
补偿模式

1)资源预留模式

在第一阶段预先留下资源,从而来保证第二阶段的执行。

比如在财务中,第一阶段将要支付的金额进行冻结,也即中间状态。在第二阶段如果是提交,则进行金额扣减,从而账号金额发生变化。如果是回滚,则将冻结进行取消,从而保证金额不会发生变化。

此时:

try阶段:完成业务检查,和预留好资源

confirm阶段:基于try执行真正的业务逻辑

cancel阶段:释放预留的资源

2)补偿模式

在实际业务中,补偿模式要用得多一些。比如下订单、减库存的时候,就会用到。

比如:下订单的时候,首先进行一个插入操作,减库存的第一阶段减少库存,并记录库存减少的数量。

第二阶段,如果是全局提交,则不用做什么,只需要将第一阶段记录的信息进行删除即可。如果是全局回滚,则进行补偿,将扣减的库存还回来即可。

此时:

try阶段: 执行真正的业务逻辑

confirm阶段: 将额外的记录信息删除即可

cancel阶段: 根据额外的记录信息对try发生失败进行补偿

二、Seata中,业务系统注解的增强

和AT模式一样,TCC模式的事务的发起方都是需要添加@GlobalTransactional这个注解的。因此我们可以看着这里有两个拦截器值得我们去关注,TccActionInterceptor与GlobalTransactionalInterceptor全局事务拦截器。因为它们是进行业务系统分布式事务增强的源头。

TCC模式的实现可以根据注解TwoPhaseBusinessAction找到对应的拦截器TccActionInterceptor。从而进一步找到

TccActionInterceptor implements MethodInterceptor, ConfigurationChangeListener, Ordered

在往上看,可以看到其是基于AOP实现的增强:

GlobalTransactionScanner extends AbstractAutoProxyCreator
        implements ConfigurationChangeListener, InitializingBean, ApplicationContextAware, DisposableBean {

因为其实现了AbstractAutoProxyCreator抽象自动代理创建器和InitializingBean、ApplicationContextAware。因此我们可以看到它的重要方法:wrapIfNecessary。这个如果看过Spring的AOP源码的话,就会知道这个方法是进行AOP增强的必经之路。

if (TCCBeanParserUtils.isTccAutoProxy(bean, beanName, applicationContext)) {
                    // init tcc fence clean task if enable useTccFence
                    TCCBeanParserUtils.initTccFenceCleanTask(TCCBeanParserUtils.getRemotingDesc(beanName), applicationContext);
                    //TCC interceptor, proxy bean of sofa:reference/dubbo:reference, and LocalTCC
                    interceptor = new TccActionInterceptor(TCCBeanParserUtils.getRemotingDesc(beanName));
                    ConfigurationCache.addConfigListener(ConfigurationKeys.DISABLE_GLOBAL_TRANSACTION,
                            (ConfigurationChangeListener)interceptor);
                }else {
                   //接口和实现都能够找到拦截器,进行全局事务拦截
                    Class<?> serviceInterface = SpringProxyUtils.findTargetClass(bean);
                    Class<?>[] interfacesIfJdk = SpringProxyUtils.findInterfaces(bean);
                    if (!existsAnnotation(serviceInterface)
                            && !existsAnnotation(interfacesIfJdk)) {
                        return bean;
                    }
                    if (globalTransactionalInterceptor == null) {
                        globalTransactionalInterceptor = new GlobalTransactionalInterceptor(failureHandlerHook);
                        ConfigurationCache.addConfigListener(
                                ConfigurationKeys.DISABLE_GLOBAL_TRANSACTION,
                                (ConfigurationChangeListener)globalTransactionalInterceptor);
                    }
                    interceptor = globalTransactionalInterceptor;
                }
                 //进行aop的增强
                if (!AopUtils.isAopProxy(bean)) {
                    bean = super.wrapIfNecessary(bean, beanName, cacheKey);
                } else {
                    AdvisedSupport advised = SpringProxyUtils.getAdvisedSupport(bean);
                    Advisor[] advisor = buildAdvisors(beanName, getAdvicesAndAdvisorsForBean(null, null, null));
                    int pos;
                    for (Advisor avr : advisor) {
                        // Find the position based on the advisor's order, and add to advisors by pos
                        pos = findAddSeataAdvisorPosition(advised, avr);
                        advised.addAdvisor(pos, avr);
                    }
                }
                PROXYED_SET.add(beanName);
                return bean;
            }

是否是TCC自动代理,如果是则初始化TCC清理Task,同时创建TCCActionInterceptor对象,添加到配置缓存中,以便可以扫描到带有TwoPhaseBusinessAction的class。

同时我们注意到其实TCC和AT模式一样,事务的发起方都是需要添加@GlobalTransactional这个注解的。因此还有一个拦截器GlobalTransactionalInterceptor全局事务拦截器。

下一篇我们来看看,TccActionInterceptor和GlobalTransactionalInterceptor都做了哪些事情。因为它们必须要和两个注解建立联系,才能发挥作用。

目录
相关文章
|
3月前
|
自然语言处理 监控 Dubbo
Seata常见问题之使用tcc模式配置yml如何解决
Seata 是一个开源的分布式事务解决方案,旨在提供高效且简单的事务协调机制,以解决微服务架构下跨服务调用(分布式场景)的一致性问题。以下是Seata常见问题的一个合集
138 4
|
2月前
|
Apache 开发者
Apache Seata 如何解决 TCC 模式的幂等、悬挂和空回滚问题
【6月更文挑战第8天】Apache Seata 是一款分布式事务框架,解决TCC模式下的幂等、悬挂和空回滚问题。通过记录事务状态处理幂等,设置超时机制避免悬挂,明确标记Try操作成功来处理空回滚。Seata 提供丰富配置和管理功能,确保分布式事务的可靠性和效率,支持复杂事务处理场景,为企业业务发展提供支持。
106 7
|
3月前
|
存储 Java Nacos
Seata常见问题之xa模式出现错误xid is not valid如何解决
Seata 是一个开源的分布式事务解决方案,旨在提供高效且简单的事务协调机制,以解决微服务架构下跨服务调用(分布式场景)的一致性问题。以下是Seata常见问题的一个合集
124 4
|
3月前
|
NoSQL Java 数据库
Seata常见问题之xa模式下插入一条数据再更新这条数据会报错如何解决
Seata 是一个开源的分布式事务解决方案,旨在提供高效且简单的事务协调机制,以解决微服务架构下跨服务调用(分布式场景)的一致性问题。以下是Seata常见问题的一个合集
154 2
|
3月前
|
Java 关系型数据库 微服务
Seata常见问题之项目一直启动不成功如何解决
Seata 是一个开源的分布式事务解决方案,旨在提供高效且简单的事务协调机制,以解决微服务架构下跨服务调用(分布式场景)的一致性问题。以下是Seata常见问题的一个合集
353 0
|
3月前
|
Nacos 数据库
分布式事务解决方案Seata
分布式事务解决方案Seata
72 1
|
3月前
|
SQL 关系型数据库 数据库
学习分布式事务Seata看这一篇就够了,建议收藏
学习分布式事务Seata看这一篇就够了,建议收藏
|
3月前
|
存储 关系型数据库 MySQL
基于Seata实现分布式事务
通过以上步骤,你可以使用 Seata 实现分布式事务,确保在微服务架构中的事务一致性。Seata 支持多种语言和框架,能够满足不同业务场景的需求。欢迎关注威哥爱编程,一起学习成长。
|
10天前
|
关系型数据库 MySQL 数据库
SpringCloud2023中使用Seata解决分布式事务
对于分布式系统而言,需要保证分布式系统中的数据一致性,保证数据在子系统中始终保持一致,避免业务出现问题。分布式系统中对数据的操作要么一起成功,要么一起失败,必须是一个整体性的事务。Seata简化了这个使用过程。
17 2
|
19天前
|
Java 关系型数据库 MySQL
(二十七)舞动手指速写一个Seata-XA框架解决棘手的分布式事务问题
相信大家对于事务问题都不陌生,在之前《MySQL事务篇》中曾详解过MySQL的事务机制,在传统的单库环境下开发,咱们可依赖于MySQL所提供的事务机制,来确保单个事务内的一组操作,要么全部执行成功,要么全部执行失败。

热门文章

最新文章