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

Seata介绍

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

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

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

单机事务场景

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

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

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

分布式事务场景

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

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

Seata解决方案

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

Seata框架包含三部分内容:

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

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

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

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. 帐户微服务:从用户帐户中扣除余额;

服务接口的定义

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

  • 仓储服务

    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框架,具体工作原理在后续文章中详细介绍。

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

参考文档

Seata官方文档

本文最先发布至微信公众号,版权所有,禁止转载!

分布式事务(七)之Seata简介的更多相关文章

  1. 分析 5种分布式事务方案,还是选了阿里的 Seata(原理 + 实战)

    好长时间没发文了,最近着实是有点忙,当爹的第 43 天,身心疲惫.这又赶上年底,公司冲 KPI 强制技术部加班到十点,晚上孩子隔两三个小时一醒,基本没睡囫囵觉的机会,天天处于迷糊的状态,孩子还时不时起 ...

  2. 微服务痛点-基于Dubbo + Seata的分布式事务(AT)模式

    前言 Seata 是一款开源的分布式事务解决方案,致力于提供高性能和简单易用的分布式事务服务.Seata 将为用户提供了 AT.TCC.SAGA 和 XA 事务模式,为用户打造一站式的分布式解决方案. ...

  3. 微服务痛点-基于Dubbo + Seata的分布式事务(TCC模式)

    前言 Seata 是一款开源的分布式事务解决方案,致力于提供高性能和简单易用的分布式事务服务.Seata 将为用户提供了 AT.TCC.SAGA 和 XA 事务模式,为用户打造一站式的分布式解决方案. ...

  4. 对比7种分布式事务方案,还是偏爱阿里开源的Seata,真香!(原理+实战)

    前言 这是<Spring Cloud 进阶>专栏的第六篇文章,往期文章如下: 五十五张图告诉你微服务的灵魂摆渡者Nacos究竟有多强? openFeign夺命连环9问,这谁受得了? 阿里面 ...

  5. Spring Cloud Alibaba 使用Seata解决分布式事务

    为什么会产生分布式事务? 随着业务的快速发展,网站系统往往由单体架构逐渐演变为分布式.微服务架构,而对于数据库则由单机数据库架构向分布式数据库架构转变.此时,我们会将一个大的应用系统拆分为多个可以独立 ...

  6. Spring Cloud Alibaba分布式事务组件 seata 详解(小白都能看懂)

    一,什么是事务(本地事务)? 指作为单个逻辑工作单元执行的一系列操作,要么完全地执行,要么完全地不执行. 简单的说,事务就是并发控制的单位,是用户定义的一个操作序列.      而一个逻辑工作单元要成 ...

  7. 浅谈,seata在使用feign-url通过域名调用时分布式事务不生效的问题及解决

    浅谈,seata在使用feign-url通过域名调用时分布式事务不生效的问题及解决 ​ 在前几个月时,我们项目出现了分布式事务的问题,那么什么是分布式事务问题呢,简单的说,我们有俩服务A和B,它们对应 ...

  8. Seata–分布式事务

    10.1 分布式事务基础 10.1.1 事务 事务指的就是一个操作单元,在这个操作单元中的所有操作最终要保持一致的行为,要么所有操作都成功,要么所有的操作都被撤销.简单地说,事务提供一种"要 ...

  9. 微服务架构 | 11.1 整合 Seata AT 模式实现分布式事务

    目录 前言 1. Seata 基础知识 1.1 Seata 的 AT 模式 1.2 Seata AT 模式的工作流程 1.3 Seata 服务端的存储模式 1.4 Seata 与 Spring Clo ...

随机推荐

  1. 编程模仿MySql客服端

    写在前面 通过自己编写的Java代码程序,去模仿实现MySql客服端的简单功能,最终以控制台操作,很像在Dos窗口通过命令操作MySql数据库. 关键问题 在编写过程中遇到的一些小问题和一些值得留心注 ...

  2. Unity——EasyTouch摇杆插件使用

    EasyTouch摇杆插件使用 Demo展示 双指缩放在电脑端无法掩饰,竖屏将就看看吧: 插件名叫EasyTouch,有需要给我留言,不想开仓库传了: 创建摇杆点这里: 初始化 On_Joystick ...

  3. noj加1乘2平方

    广度优先搜索典例 00 题目 描述: 最简单的队列的使用#include <iostream>#include <queue>using namespace std;queue ...

  4. 【c++ Prime 学习笔记】第14章 重载运算与类型转换

    14.1 基本概念 重载的运算符是特殊的函数:名字由关键字operator后接要定义的算符共同组成,也有返回类型.参数列表.函数体. 重载运算符函数的参数量与该算符作用的运算对象数量一样多 除重载调用 ...

  5. Java:并发笔记-01

    Java:并发笔记-01 说明:这是看了 bilibili 上 黑马程序员 的课程 java并发编程 后做的笔记 1. 进程与线程 本章内容 进程和线程的概念 并行和并发的概念 线程基本应用 1.1 ...

  6. 基于websocket实现的一个简单的聊天室

    本文是基于websocket写的一个简单的聊天室的例子,可以实现简单的群聊和私聊.是基于websocket的注解方式编写的.(有一个小的缺陷,如果用户名是中文,会乱码,不知如何处理,如有人知道,请告知 ...

  7. cadence 技巧

    pcb中如何选中完整的一条网络? 1 edit  properties  右边 find nets 2 cadence 选中不同的网络高亮 display--->assign color在opt ...

  8. Java:检查异常与未检查异常

    一.异常的介绍 Throwable 是 Java 中所有错误和异常的超类.Java 虚拟机仅抛出属于此类(或其子类之一)的实例对象,或者是 throw 语句也可以抛出该对象.同样,catch 子句中的 ...

  9. linux切换shell

    1. $SHELL这一环境变量用于保存当前用户使用的shell,所以我们可以输出$SHELL来查看当前使用的shell是什么: 2. 查看/etc/shells文件,可以看到当前系统中安装的有效的sh ...

  10. 开发笔记-----Ajax 基础使用

    一.GET 方式的用法: 1 <!--html --> 2 <div class="layui-form"> 3 <div class="l ...