在微服务架构下,虽然我们会尽量避免分布式事务,但是只要业务复杂的情况下这是一个绕不开的问题,如何保证业务数据一致性呢?本文主要介绍同步场景下使用 Seata
的 AT模式
来解决一致性问题。
Seata
是 阿里巴巴 开源的 一站式分布式事务解决方案 中间件,以 高效 并且对业务 0 侵入 的方式,解决 微服务 场景下面临的分布式事务问题
整体事务逻辑是基于 两阶段提交 的模型,核心概念包括以下3个角色:
目前的 Seata
有两种模式可使用分别对应不同业务场景
ACID JDBC
一个典型的分布式事务过程:
TM
向 TC
申请开启一个全局事务,全局事务创建成功并生成一个全局唯一的 XID
。 XID
在微服务调用链路的上下文中传播。 RM
向 TC
注册分支事务,将其纳入 XID 对应全局事务的管辖。 TM
向 TC
发起针对 XID
的全局提交或回滚决议。 TC
调度 XID
下管辖的全部分支事务完成提交或回滚请求。 该模式逻辑类似 TCC
,需要 自定义实现 prepare
、 commit
和 rollback
的逻辑,适合 非关系型数据库 的场景
模拟一个简单的用户下单场景,4个子工程分别是 Bussiness(事务发起者) 、 Order(创建订单) 、 Storage(扣减库存) 和 Account(扣减账户余额)
Discover
注册、 Config
配置和 Store
存储模块默认都是使用 file
只能适用于单机,我们安装的时候分别改成使用 nacos
和 Mysql
以支持server端集群
https://github.com/seata/seata/releases
注册中心和配置中心默认是 file
这里改为 nacos
;设置 registry 和 config 节点中的 type
为 nacos
,修改 serverAddr
为你的 nacos
节点地址。
registry { type = "nacos" nacos { serverAddr = "192.168.28.130" namespace = "public" cluster = "default" } } config { type = "nacos" nacos { serverAddr = "192.168.28.130" namespace = "public" cluster = "default" } }
修改 service.vgroup_mapping 为自己应用对应的名称;如果有多个服务,添加相应的配置
默认组名为 ${spring.application.name}-fescar-service-group
,可通过 spring.cloud.alibaba.seata.tx-service-group
配置修改
修改 store.mode 为 db
,并修改数据库相关配置
cd conf sh nacos-config.sh 192.168.28.130
成功后在 nacos
的配置列表中能看到 seata
的相关配置
执行 conf/db_store.sql
中的脚本
sh bin/seata-server.sh -p 8091 -h 192.168.28.130
执行脚本 seata-demo.sql
需在业务相关的数据库中添加 undo_log 表,用于保存需要回滚的数据
直接把 seata-server 中的 registry.conf
复制到每个服务中去即可,不需要修改
demo中的每个服务各自修改配置文件
Seata
是通过代理数据源实现分布式事务,所以需要配置 io.seata.rm.datasource.DataSourceProxy
的 Bean
,且是 @Primary
默认的数据源,否则事务不会回滚,无法实现分布式事务
public class DataSourceProxyConfig { @Bean @ConfigurationProperties(prefix = "spring.datasource") public DruidDataSource druidDataSource() { return new DruidDataSource(); } @Primary @Bean public DataSourceProxy dataSourceProxy(DruidDataSource druidDataSource) { return new DataSourceProxy(druidDataSource); } }
因为使用了mybatis的starter所以需要排除 DataSourceAutoConfiguration
,不然会产生循环依赖
@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class})
事务发起者 business-service
添加 @GlobalTransactional
注解
@GlobalTransactional public void placeOrder(String userId) { ...... }
提供两个接口测试
https://gitee.com/zlt2000/microservices-platform/tree/master/zlt-demo/seata-demo