enchanter1cl / DTX-Seata-AT

微服务整合 Seata AT 模式解决分布式事务问题

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

erato-cloud-mall-newbee

技术选型

seata-server1.4.2

nacos-server2.0.2

分布式事务问题演示

img.png

订单生成的详细过程如下:

  • 首先是检查是否包含已下架商品,如果有则抛出一个异常,无下架商品则继续后续流程
  • 继续判断商品数据和商品库存,如果商品数据有误或者商品库存不足也会抛出异常,一切正常则继续后续流程
  • 对象的非空判断
  • 生成订单后,购物项数据需要删除,这里调用 ShoppingCartItemMapper.deleteBatch() 方法将这些数据批量删除
  • 更新商品库存记录
  • 判断订单价格,如果所有购物项加起来的数据为 0 或者小于 0 则不继续生成订单
  • 生成订单号并封装 MallOrder 对象,保存订单记录到数据库中
  • 封装订单项数据并保存订单项数据到数据库中
  • 生成订单收货地址快照,并保存至数据库
  • 返回订单号

简化后的方法代码:

public class MallOrderServiceImpl {

    @Transactional
    public Boolean saveOrder(int cartId) {
        // 调用购物车服务-获取即将操作的goods_id
        int goodsId = newBeeShopCartDemoService.getGoodsId(cartId);
        // 调用商品服务-减库存
        Boolean goodsResult = newBeeGoodsDemoService.deStock(goodsId);
        // 调用购物车服务-删除当前购物车数据
        Boolean cartResult = newBeeShopCartDemoService.deleteItem(cartId);
        // 执行下单逻辑
        if (goodsResult && cartResult) {
            // 向订单表中新增一条记录
            int orderResult = jdbcTemplate.update("insert into tb_order(`cart_id`) value (\"" + cartId + "\")");
            // 此处出现了异常
            //int i=1/0;
            if (orderResult > 0) {
                return true;
            }
            return false;
        }
        return false;
    }
}

该方法涉及到的表有订单表、订单项表、购物项表、商品表、用户信息表、用户收货地址表。只要其中任何一个步骤没有通过验证或者在任意一行代码中抛出异常,那么涉及到这些表的操作都会回滚。这就是数据库事务的特点,要么完全地执行,要么完全地不执行。

微服务场景下,这些表不在一个数据库中,没有统一的事务管理器,会发生什么呢?

img_1.png

Seata **

在跨数据库、跨分区、跨服务情况下,如果要保证一个方法中的 SQL 操作全部成功或者全部失败,则有些复杂。

当然,分布式事务这种问题存在已久,所以业内也有很多分布式事务的应对方案,比如常见的二阶段提交(2PC)、三阶段提交(3PC)、TCC方案和最终一致性方案。由于篇幅限制,这里就不进行拓展了,读者可以自行搜索上述这些关键词来学习。这些方案都是实现原理,算是一种顶层设计,如果要实际的应用到项目中,则需要使用这些方案所对应的实际软件产品。在 Java 开源领域能够提供分布式事务解决方案的软件中,比较有代表性的有 Sharding JDBC、Atomikos、MyCat、Alibaba Seata 等等。

在这些落地方案中,不管是基于长事务的方案还是基于消息通知的方案,都是对跨数据库事务的进一步抽象,形成一个新的事务概念。未加入分布式事务方案时,跨数据库的事务一致性是无法解决的,因为每个数据库都是独立的,其中的事务管理器也是独立的,互不干涉、互不影响。而上述的这些分布式事务解决方案,则是在数据库之外引入了一个新的角色,对整个流程进行进一步的抽象,在其中加入了一个协调者(或者说管理者),此时就存在一个第三方的管理者来协调多个数据库事务,如下图所示。

img_1.png

Seata 就是一个跨数据库事务的第三方“管理者”,它把跨数据库、跨服务的分布式事务场景变成了一个拥有多个分支事务的全局事务。

只有所有的分支事务都成功了,全局事务才算成功(Commit)。只要有一个分支事务失败了,则全部分支事务失败,所有的分支事务全部都会回滚(Rollback)。

建库建表准备

如果想要整合Seata来解决分布式问题,还需要在每个微服务实例所依赖的数据库中创建一张名称为undo_log的表。比如本库三个微服务实例,需要在商品服务的数据库、购物车服务的数据库和订单服务的数据库中各自新建一张 undo_log 表。具体的建表语句 Seata 官方已经提供了,地址如下:

https://github.com/seata/seata/tree/develop/script/client

在该目录下有多个文件夹,分别是 at/db、saga/db、tcc/db。因为 Seata 为开发者们提供了多种分布式事务的处理方式,比如 AT、TCC、SAGA,在选择不同的处理方案时,就需要引入不同的建表语句。

本库的编码中使用的 Seata 方案是 AT 模式。AT 模式是 Seata 官方比较推荐的一套分布式事务解决方案,因为这种方式比较简单、对业务侵入低、不需要改动具体的业务代码,添加一个注解再添加几行配置项即可整合 Seata 来解决分布式事务,非常方便。所以,需要引入的 undo_log 表的文件地址如下:

https://github.com/seata/seata/blob/develop/script/client/at/db/mysql.sql

最终的建表SQL内容见 /sql/undo_log.sql

通过 undo_log 表的建表字段可知,该表会存储全局事务和分支事务的id信息、回滚数据、执行状态等等信息,如果全局事务失败,就需要依次回滚所有分支事务,需要执行的内容就保存在这个表里。所以,undo_log 这个表需要创建在各个微服务实例下的数据库中。比如本demo,就要在 test_distribution_cart_db 数据库、test_distribution_goods_db 数据库和 test_distribution_order_db 数据库中依次创建这个表,建表成功后如下图所示。

img.png

然后添加与Seata相关的配置项。官方提供的配置文件可供参考,地址如下:

https://github.com/seata/seata/tree/develop/script/client/spring

这里有.properties和.yml两个格式的配置文件,根据自身项目配置文件的格式选择即可。主要增加的配置项如下:

  • seata.enabled:是否开启自动配置
  • seata.application-id:当前Seata客户端的应用名称
  • seata.tx-service-group:事务分组
  • seata.registry.type:服务中心的类型(本课程选择的是Nacos)
  • seata.registry.nacos.*:与Nacos相关的配置信息

新版本 Seata 已可以自动代理数据源对象。Seata 会创建 Druid 数据源并注册到 Spring 的 IOC 容器中,然后使用它来生成 DataSourceProxy 对象并注册到Spring的IOC容器中

About

微服务整合 Seata AT 模式解决分布式事务问题


Languages

Language:Java 100.0%