Spring Cloud同步场景分布式事务怎样做?试试Seata

作者简介:
陶陶老师
10年后端工作经验,
专注Java、SpringBoot、SpringCloud、分布式系统/微服务、中间件等领域。
公众号:陶陶技术笔记

一、概述
在微服务架构下,虽然我们会尽量避免分布式事务,但是只要业务复杂的情况下这是一个绕不开的问题,如何保证业务数据一致性呢?本文主要介绍同步场景下使用Seata的AT模式来解决一致性问题。

Seata是 阿里巴巴 开源的 一站式分布式事务解决方案 中间件,以 高效 并且对业务 0 侵入 的方式,解决 微服务 场景下面临的分布式事务问题

二、Seata介绍
整体事务逻辑是基于 两阶段提交 的模型,核心概念包括以下3个角色:

TM:事务的发起者。用来告诉 TC,全局事务的开始,提交,回滚。

RM:具体的事务资源,每一个 RM 都会作为一个分支事务注册在 TC。

TC:事务的协调者seata-server,用于接收我们的事务的注册,提交和回滚。

目前的Seata有两种模式可使用分别对应不同业务场景

2.1. AT模式
该模式适合的场景:

基于支持本地 ACID 事务的关系型数据库。

Java 应用,通过 JDBC 访问数据库。

![`QHAF1B6)LT]E_ZWMNH3N%G.png](https://upload-images.jianshu.io/upload_images/5954723-07a9cb25134579ef.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

一个典型的分布式事务过程:

TM 向 TC 申请开启一个全局事务,全局事务创建成功并生成一个全局唯一的 XID。

XID 在微服务调用链路的上下文中传播。

RM 向 TC 注册分支事务,将其纳入 XID 对应全局事务的管辖。

TM 向 TC 发起针对 XID 的全局提交或回滚决议。

TC 调度 XID 下管辖的全部分支事务完成提交或回滚请求。

2.2. MT模式
该模式逻辑类似TCC,需要 自定义实现 prepare、commit和rollback的逻辑,适合 非关系型数据库 的场景
![6YO9Q{0R_EK$S5LA1EGN.png

三、Seata场景样例
模拟一个简单的用户下单场景,4个子工程分别是 Bussiness(事务发起者)、Order(创建订单)、Storage(扣减库存) 和 Account(扣减账户余额)

Q@BLE82TA6WC{Y4)2CV%AVS.png

3.1. 部署Seata的Server端

![XX}NRSRV]MW}0IJO(ZDWEUP.png](https://upload-images.jianshu.io/upload_images/5954723-6338e91e04662a50.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

Discover注册、Config配置和Store存储模块默认都是使用file只能适用于单机,我们安装的时候分别改成使用nacos和Mysql以支持server端集群

3.1.1. 下载最新版本并解压
https://github.com/seata/seata/releases

3.1.2. 修改 conf/registry.conf 配置
注册中心和配置中心默认是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"
}
}

3.1.3. 修改 conf/nacos-config.txt配置

![7%~V2FGU62DFBOG]TBN3S5N(1).png](https://upload-images.jianshu.io/upload_images/5954723-e7fdca45637c63cc.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

修改 service.vgroup_mapping 为自己应用对应的名称;如果有多个服务,添加相应的配置

默认组名为${spring.application.name}-fescar-service-group,可通过spring.cloud.alibaba.seata.tx-service-group配置修改

修改 store.mode 为db,并修改数据库相关配置

3.1.4. 初始化seata的nacos配置

cd conf
sh nacos-config.sh 192.168.28.130

成功后在nacos的配置列表中能看到seata的相关配置

3.1.5. 初始化数据库
执行conf/db_store.sql中的脚本

3.1.6. 启动seata-server

sh bin/seata-server.sh -p 8091 -h 192.168.28.130

3.2. 应用配置
3.2.1. 初始化数据库
执行脚本 seata-demo.sql

CREATE DATABASE seata-demo DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
USE seata-demo;

--==========================回滚日志表==========================
-- 此脚本必须初始化在你当前的业务数据库中,用于AT 模式XID记录。与server端无关(注:业务数据库)
drop table undo_log;
CREATE TABLE undo_log (
id bigint(20) NOT NULL AUTO_INCREMENT,
branch_id bigint(20) NOT NULL,
xid varchar(100) NOT NULL,
context varchar(128) NOT NULL,
rollback_info longblob NOT NULL,
log_status int(11) NOT NULL,
log_created datetime NOT NULL,
log_modified datetime NOT NULL,
ext varchar(100) DEFAULT NULL,
PRIMARY KEY (id),
UNIQUE KEY ux_undo_log (xid,branch_id)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

--==========================业务模拟表==========================
DROP TABLE IF EXISTS storage_tbl;
CREATE TABLE storage_tbl (
id int(11) NOT NULL AUTO_INCREMENT,
commodity_code varchar(255) DEFAULT NULL,
count int(11) DEFAULT 0,
PRIMARY KEY (id),
UNIQUE KEY (commodity_code)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

DROP TABLE IF EXISTS order_tbl;
CREATE TABLE order_tbl (
id int(11) NOT NULL AUTO_INCREMENT,
user_id varchar(255) DEFAULT NULL,
commodity_code varchar(255) DEFAULT NULL,
count int(11) DEFAULT 0,
money int(11) DEFAULT 0,
PRIMARY KEY (id)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

DROP TABLE IF EXISTS account_tbl;
CREATE TABLE account_tbl (
id int(11) NOT NULL AUTO_INCREMENT,
user_id varchar(255) DEFAULT NULL,
money int(11) DEFAULT 0,
PRIMARY KEY (id)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

-- 初始化库存模拟数据
INSERT INTO storage_tbl (id, commodity_code, count) VALUES (1, 'P001', 9999999);
INSERT INTO account_tbl (id, user_id, money) VALUES ('1', 'U001', 10000);

需在业务相关的数据库中添加 undo_log 表,用于保存需要回滚的数据

3.2.2. 添加registry.conf配置
直接把 seata-server 中的registry.conf复制到每个服务中去即可,不需要修改

D@H)TTRQU3123AE[~)]UN_T(1).png

3.2.3. 修改配置
demo中的每个服务各自修改配置文件

bootstrap.yml 修改nacos地址

application.yml 修改数据库配置

3.2.4. 配置数据源代理
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})

3.2.5. 事务发起者添加全局事务注解
事务发起者 business-service 添加 @GlobalTransactional 注解

@GlobalTransactional
public void placeOrder(String userId) {
......
}

3.3. 测试
提供两个接口测试

事务成功:扣除库存成功 > 创建订单成功 > 扣减账户余额成功
http://localhost:9090/placeOrder

事务失败:扣除库存成功 > 创建订单成功 > 扣减账户余额失败,事务回滚
http://localhost:9090/placeOrderFallBack

3.4. demo地址
https://gitee.com/zlt2000/microservices-platform/tree/master/zlt-demo/seata-demo

作者简介:
陶陶老师
10年后端工作经验,
专注Java、SpringBoot、SpringCloud、分布式系统/微服务、中间件等领域。
公众号:陶陶技术笔记

本文已经获得陶陶老师授权转发,其他人若有兴趣转载,请直接联系作者授权。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容

  • 使用步骤: 一、安装 下载seata-server:https://github.com/seata/seata/...
    窦海超阅读 1,620评论 0 2
  • 3. 分布式事务解决方案之2PC(两阶段提交) 针对不同的分布式场景业界常见的解决方案有2PC、TCC、可靠消息最...
    小小一技术驿站阅读 1,733评论 2 2
  • 11一21,2017年6月10日,让爱醒来践行 1,恭敬一切缘,感恩一切缘。出发东莞,路上无论遇到谁,都能心怀恭敬...
    为_福慧阅读 187评论 0 0
  • 今天,我带儿子和女儿回父母家,我想要看他们。之所以想见他们,一方面是担心爸爸的伤情,另一方面因为上次书写后发现,我...
    慢慢公主阅读 389评论 0 4
  • 2017年最后一张小画,也是第一张挤娃娃。从来没有画过这种画,感觉挺好玩的。在不舍中告别了2017,不问前路漫漫,...
    M有如果阅读 291评论 8 15