分布式事务(七)之Seata简介

柯里蝉翼
• 阅读 1336

在前面的文章中,我们介绍了分布式事务的概念以及一些解决方案。fenSeata是一款开源的分布式事务解决方案,致力于提供高性能和简单易用的分布式事务服务。Seata将为用户提供了AT、TCC、SAGA和XA事务模式,为用户打造一站式的分布式解决方案。

Seata介绍

本文以一个用户下单购买商品的系统为例,介绍开源框架Seata的原理和使用,下单该系统涉及三部分服务:

  1. 仓储服务:对给定的商品扣除仓储数量;
  2. 订单服务:根据采购需求创建订单;
  3. 帐户服务:从用户帐户中扣除余额;

分布式事务的主要作用是保证微服务情况下用户下单过程中数据的一致性。这里的一致性可以这样理解:不会出现用户余额扣除成功,但是仓储和订单相关操作失败的场景,三者要么同时成功,要么同时失败。

单机事务场景

如果用户下单购买商品涉及到的服务都在一个传统的单机服务中,三部分服务可以共享同一个数据库实例。这种情况下,我们可以通过本地事务的一致性保证仓储/订单/账户三者之间数据的一致性。

分布式事务(七)之Seata简介

如上图所示,在单机服务中,三部分内容共用同一个数据库实例,所以我们只需要本地事务就可以解决数据的一致性,以Spring框架为例,我们只需要在方法上添加@Transaction注解就可以实现整个购买流程中数据的一致性:

@Transaction
public void purchase(){
    doStoreBusiness();
    doOrderBusiness();
    doAccountBusiness();
}

分布式事务场景

在微服务框架中,仓储/订单/账户服务部署在不同的服务器上,使用不同的数据库实例,与单机模式完全不同。单机模式中的事务通常要求事务涉及的数据源为同一个,并且事务涉及的数据库操作在同一个数据库链接中,分布式情况下显然不满足条件。

所以在分布式场景如何保证数据的一致性呢?这就是Seata需要解决的问题了。

分布式事务(七)之Seata简介

Seata解决方案

Seata是用于解决分布式事务的开源框架,其内部有关于分布式事务的定义如下:分布式事务是由多个分支事务组成的全局事务,其中每个分支事务都是本地事务的形式。

分布式事务(七)之Seata简介

Seata框架包含三部分内容:

  1. 事务协调器(Transaction Coordinator,TC):维护全局事务和分支事务的状态,进行全局事务提交或全局事务回滚;
  2. 事务管理器(Transaction Manager,TM):定义全局事务,开启全局事务、提交全局事务或回滚全局事务;
  3. 资源管理器(Resource Manager,RM):管理分支事务中的资源,向事务管理器注册分支事务和并报告分支事务的状态,负责分支事务提交或回滚;

分布式事务(七)之Seata简介

一个典型的seata分布式事务的流程如下:

  1. TM向TC发出开启全局事务请求,TC生成全局事务的唯一标识XID,设此处的全局事务为T1;
  2. 在全局事务T1的各个流程中,XID会作为事务的标识在微服务之间流转;
  3. RM向TC注册本地事务,注册的本地事务的会作为全局事务T1的分支事务;
  4. TM可以请求TC控制全局事务T1提交或全局事务T1回滚;
  5. TC可以请求全局事务T1下的所有分支事务提交或回滚;

分布式事务(七)之Seata简介

Seata历史

阿里巴巴:

  • TXC:淘宝交易系统的分布式事务框架,阿里巴巴中间件团队自2014年开始启动该项目,用于解决应用架构从单一服务向微服务转变所带来的分布式事务问题;
  • GTS:全球交易服务。TXC作为阿里云中间件产品,新名称GTS于2016年发布;
  • Fescar:2019年开始了基于TXC/GTS的开源项目Fescar,用于开源项目社区发展;

蚂蚁金服:

  • XTS:扩展事务服务。蚂蚁金服中间件团队自2007年开始开发分布式事务中间件,该中间件在蚂蚁金服中得到广泛应用,解决了跨数据库和服务的数据一致性问题。
  • DTX:分布式事务扩展。自2013年以来,XTS已在蚂蚁金融云上发布,名称为DTX;

Seata社区:

  • Seata:简单的可扩展分布式事务解决方案,蚂蚁金服将Fedscar改名为Seata并开源,使其成为一个中立、开放的分布式事务社区。

Seata Maven依赖

<seata.version>1.4.2</seata.version>

<dependency>
    <groupId>io.seata</groupId>
    <artifactId>seata-all</artifactId>
    <version>${seata.version}</version>
</dependency>

Seata案例

以上文中的用户下单购买商品的系统为例,展示Seata的使用方式:

  1. 仓储微服务:对给定的商品扣除仓储数量;
  2. 订单微服务:根据采购需求创建订单;
  3. 帐户微服务:从用户帐户中扣除余额;

分布式事务(七)之Seata简介

服务接口的定义

对于三个微服务,我们用三个接口来抽象其内部的逻辑:

  • 仓储服务

    public interface StorageService {
    
          /**
          * 扣除存储数量
          */
          void deduct(String commodityCode, int count);
      }
  • 订单服务

      public interface OrderService {
          /**
          * 创建订单
          */
          Order create(String userId, String commodityCode, int orderCount);
      }
  • 帐户服务

    public interface AccountService {
    
        /**
        * 从用户账户中借出
        */
        void debit(String userId, int money);
    }

主要业务逻辑

对于用户下单购买商品的逻辑,我们用以下代码来实现:

  • 主要业务逻辑

      public class BusinessServiceImpl implements BusinessService {
    
          private StorageService storageService;
    
          private OrderService orderService;
    
          /**
          * 采购
          */
          public void purchase(String userId, String commodityCode, int orderCount) {
    
              storageService.deduct(commodityCode, orderCount);
    
              orderService.create(userId, commodityCode, orderCount);
          }
      }
  • 订单服务业务逻辑

      public class OrderServiceImpl implements OrderService {
    
          private OrderDAO orderDAO;
    
          private AccountService accountService;
    
          public Order create(String userId, String commodityCode, int orderCount) {
    
              int orderMoney = calculate(commodityCode, orderCount);
    
              accountService.debit(userId, orderMoney);
    
              Order order = new Order();
              order.userId = userId;
              order.commodityCode = commodityCode;
              order.count = orderCount;
              order.money = orderMoney;
    
              // INSERT INTO orders ...
              return orderDAO.insert(order);
          }
      }

引入Seata

在服务中引入Seata服务之后,我们只需要在分布式事务最外层的方法上添加分布式事务注解@GlobalTransactional

@GlobalTransactional
public void purchase(String userId, String commodityCode, int orderCount) {
    // ......
}

添加注解之后,在执行业务逻辑之前,Seata会先生成全局事务ID,并且在调用其它服务时,会在请求中携带全局事务ID。如果其它微服务也添加了Seata依赖,这些微服务会获取全局事务ID,并且参与到全局事务中。

本文只是简单介绍以下Seata框架,具体工作原理在后续文章中详细介绍。

分布式事务(七)之Seata简介

我是御狐神,欢迎大家关注我的微信公众号:wzm2zsd

分布式事务(七)之Seata简介

参考文档

Seata官方文档

本文最先发布至微信公众号,版权所有,禁止转载!
点赞
收藏
评论区
推荐文章
Wesley13 Wesley13
4年前
REST微服务的分布式事务实现
上一篇文章REST微服务的分布式事务实现分布式事务以及JTA介绍 中,试着带大家理解事务,然后介绍了分布式事务、它的原则和实现方式。这一部分,我们就来详细看看如何使用消息中间件来实现分布式事务。我们还是使用之前的实例,一个订票系统的购票逻辑:!(https://oscimg.oschina.net/oscnet/3582cf8efeb63070
基于Seata探寻分布式事务的实现方案
随着业务的快速发展、业务复杂度越来越高,几乎每个公司的系统都会从单体走向分布式,特别是转向微服务架构。随之而来就必然遇到分布式事务这个难题,这篇文章通过seata框架总结了分布式事务的几种解决方案。
一种基于柔性事务的分布式事务解决方案设计探究
1背景市面上常见的有,2pc/3pc、tcc、saga等常见的分布式事务解决方案,但是实际实施起来框架比较重,设计开发比较繁琐,不易于快速开发上手。本文提供一种基于柔性事务设计的简单易上手的分布式事务设计方案,用于解决常见的分布式事务常见
Stella981 Stella981
4年前
Spring Cloud整合Seata实现分布式事务
SpringCloud整合Seata分布式事务框架Seata(https://www.oschina.net/action/GoToLink?urlhttps%3A%2F%2Fgithub.com%2Fseata%2Fseata):阿里巴巴开源的一款分布式解决方案,其前身是Fescar。官
Stella981 Stella981
4年前
Bag分布式事务:对SAGA分布式事务的改进
最近打算给jSqlBox添加分布式事务功能,研究了几种分布式事务方案,对SAGA模式比较感兴趣,它是通过将多个事务隔离成多个单个事务,顺序执行(或回滚阶段倒序对冲)来完成的,但是SAGA分布式事务不能保证隔离性的问题,因为单纯的SAGA模式没有锁住资源。经考虑发现在SAGA的思路上,利用事务嵌套和全局锁,可以实现一种简单的分布式事务实现,暂时给它起个新名字叫
Stella981 Stella981
4年前
ShardingSphere x Seata,一致性更强的分布式数据库中间件
日前,分布式数据库中间件 ShardingSphere 将Seata分布式事务能力进行整合,旨在打造一致性更强的分布式数据库中间件。背景数据库领域,分布式事务的实现主要包含:两阶段的XA和BASE柔性事务。XA事务底层,依赖于具体的数据库厂商对XA两阶段提交协议的支持。通常,XA协议通过在Prepare和Co
Stella981 Stella981
4年前
Seata是什么?一文了解其实现原理
一、背景随着业务发展,单体系统逐渐无法满足业务的需求,分布式架构逐渐成为大型互联网平台首选。伴随而来的问题是,本地事务方案已经无法满足,分布式事务相关规范和框架应运而生。在这种情况下,大型厂商根据分布式事务实现规范,实现了不同的分布式框架,以简化业务开发者处理分布式事务相关工作,让开发者专注于核心业务开发。Seata就是这么一个分布式事
如来佛祖 如来佛祖
1年前
一课学透 分布式事务框架 Alibaba Seata
一课学透分布式事务框架AlibabaSeataAlibabaSeata是一个开源的分布式事务解决方案,旨在解决分布式事务的一致性问题。下面是关于Seata的详细介绍,希望能帮助你全面了解这个框架:1.什么是分布式事务?分布式系统中的分布式事务指的是跨多个独