动手实践Seata四种模式(XA、AT、TCC、SAGA)

动手实践

1、XA模式

XA模式原理

XA 规范 是 X/Open 组织定义的分布式事务处理(DTP,Distributed Transaction Processing)标准,XA 规范 描述了全局的TM与局部的RM之间的接口,几乎所有主流的数据库都对 XA 规范 提供了支持。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-imTbvx4I-1654689753280)(images/image-20220608101341216.png)]

如果有失败的就会回滚事务

seata的XA模式

seata的XA模式做了一些调整,但大体相似:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-DHYEGEK3-1654689753281)(images/image-20220608101634208.png)]

RM一阶段的工作:

  1. 注册分支事务到TC
  2. 执行分支业务sql但不提交
  3. 报告执行状态到TC

TC二阶段的工作:

  1. TC检测各分支事务执行状态
    • 如果都成功,通知所有RM提交事务
    • 如果有失败,通知所有RM回滚事务

RM二阶段的工作:

  1. 接收TC指令,提交或回滚事务
xa模式的优点:
  1. 事务的强一致性,满足ACID原则。
  2. 常用数据库都支持,实现简单,并且没有代码侵入
xa模式的缺点:
  1. 因为一阶段需要锁定数据库资源,等待二阶段结束才释放,性能较差
  2. 依赖关系型数据库实现事务

实现XA模式

Seata的starter已经完成了XA模式的自动装配,实现非常简单,步骤如下:
  1. 修改application.yml文件(每个参与事务的微服务),开启XA模式:

    # 配置seata的注册中心
    seata:
      data-source-proxy-mode: XA # 选择XA模式
    

    注意:是每一个微服务都需要

  2. 给发起全局事务的入口方法添加@GlobalTransactional注解,本例中是OrderServiceImpl中的create方法:

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ETitQLdz-1654689753282)(images/image-20220608105627606.png)]

  3. 启动所有微服务,postman进行接口测试

    先进行正确的测试,再继续错误的测试

    错误设置,购买商品超出原来剩余的商品数就会让数据库报错测试是否可以事务回滚

    注意:如果测试接口报错响应时间过长,那么就应该设置响应的时间大一点,如下图,然后重启seata

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gu1qDuVJ-1654689753282)(images/image-20220608105938074.png)]

    成功的可以查看seate的控制输出,可以看到事务回滚

    IDEA输出

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-zedLAk8N-1654689753282)(images/image-20220608110122848.png)]

    查看数据库的数据是否有被更新

2、AT模式

2.1、AT模式原来

AT模式同样是分阶段提交的事务模型,不过缺弥补了XA模型中资源锁定周期过长的缺陷。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-meFJC4pN-1654689753283)(images/image-20220608143602661.png)]

阶段一RM的工作:

  1. 注册分支事务
  2. 记录undo-log(数据快照,JSON格式)
  3. 执行业务sql并提交
  4. 报告事务状态

阶段二提交时RM的工作:

  1. 删除undo-log即可

阶段二回滚时RM的工作:

  1. 根据undo-log恢复数据到更新前
执行示例

例如,一个分支业务的SQL是这样的:update tb_account set money = money - 10 where id = 1

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-O5Fnc3BK-1654689753283)(images/image-20220608143806379.png)].

AT模式与XA模式最大的区别是什么
  1. XA模式一阶段不提交事务,锁定资源;AT模式一阶段直接提交,不锁定资源。
  2. XA模式依赖数据库机制实现回滚;AT模式利用数据快照实现数据回滚。
  3. XA模式强一致;AT模式最终一致
2.2、AT模式的脏写问题

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4zvyYKpl-1654689753284)(images/image-20220608144903373.png)]

2.3、AT模式的写隔离

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-f8JiRvCz-1654689753285)(images/image-20220608145045957.png)]

2.4、AT模式的优点
  1. 一阶段完成直接提交事务,释放数据库资源,性能比较好
  2. 利用全局锁实现读写隔离
  3. 没有代码侵入,框架自动完成回滚和提交
AT模式的缺点
  1. 两阶段之间属于软状态,属于最终一致
  2. 框架的快照功能会影响性能,但比XA模式要好很多
2.5、实现AT模式

AT模式中的快照生成、回滚等动作都是由框架自动完成,没有任何代码侵入,因此实现非常简单。

1、创建相关的数据库文件

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-sXVOEwga-1654689753285)(images/image-20220608150000377.png)]

lock_table(全局锁)表导入到TC服务关联的数据库

DROP TABLE IF EXISTS `lock_table`;
CREATE TABLE `lock_table`  (
  `row_key` varchar(128) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL,
  `xid` varchar(96) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `transaction_id` bigint(20) NULL DEFAULT NULL,
  `branch_id` bigint(20) NOT NULL,
  `resource_id` varchar(256) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `table_name` varchar(32) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `pk` varchar(36) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `gmt_create
  • 24
    点赞
  • 62
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 19
    评论
Seata支持多种分布式事务协议,其中最常用的有以下四种: 1. AT(自动补偿型):AT模式是基于数据库的存储过程实现的,它通过事务日志记录所有的数据更新操作,以便在回滚时执行相应的反向操作来达到事务回滚的目的。AT模式相对来说比较简单,但对于一些复杂的业务场景可能需要手动实现反向操作。 2. TCC(两阶段型):TCC模式是基于“预留资源”和“确认资源”的思想实现的,它将整个事务分为“Try”、“Confirm”和“Cancel”三个阶段,分别对应事务的执行、提交和回滚操作。TCC模式需要手动实现各个阶段的逻辑,相对来说比较复杂,但可以在一定程度上提高事务的并发性。 3. SAGA(补偿型):SAGA模式是基于“补偿事务”实现的,它通过记录每个事务的“补偿操作”来达到事务回滚的目的。SAGA模式相对来说比较灵活,可以适应各种复杂的业务场景,但需要手动实现每个事务的“补偿操作”,工作量比较大。 4. XA(两阶段型):XA模式是基于“两阶段提交”协议实现的,它通过协调多个事务资源的提交和回滚来达到事务一致性的目的。XA模式相对来说比较成熟,但需要支持XA协议的数据库驱动和中间件支持,同时也存在单点故障和性能瓶颈的问题。 在Seata中,可以通过配置文件来选择使用哪种分布式事务协议,例如: ```properties # AT模式 spring.cloud.alibaba.seata.tx-service-group=my_test_tx_group seata.tx-service-group=my_test_tx_group seata.enable-auto-data-source-proxy=true seata.config.nacos.config-type=Nacos seata.config.nacos.server-addr=127.0.0.1:8848 seata.config.nacos.namespace= seata.config.nacos.groupId=SEATA_GROUP seata.config.nacos.dataId=seata.yml seata.enable-auto-compensate=true # TCC模式 spring.cloud.alibaba.seata.tx-service-group=my_test_tx_group seata.tx-service-group=my_test_tx_group seata.enable-auto-data-source-proxy=true seata.config.nacos.config-type=Nacos seata.config.nacos.server-addr=127.0.0.1:8848 seata.config.nacos.namespace= seata.config.nacos.groupId=SEATA_GROUP seata.config.nacos.dataId=seata.yml seata.service.vgroup-mapping.my_test_tx_group=default seata.service.default.grouplist=127.0.0.1:8091 seata.tcc.storage-mode=db # SAGA模式 spring.cloud.alibaba.seata.tx-service-group=my_test_tx_group seata.tx-service-group=my_test_tx_group seata.enable-auto-data-source-proxy=true seata.config.nacos.config-type=Nacos seata.config.nacos.server-addr=127.0.0.1:8848 seata.config.nacos.namespace= seata.config.nacos.groupId=SEATA_GROUP seata.config.nacos.dataId=seata.yml seata.enable-auto-compensate=true seata.saga.auto-proxy-target-class=true # XA模式 spring.cloud.alibaba.seata.tx-service-group=my_test_tx_group seata.tx-service-group=my_test_tx_group seata.enable-auto-data-source-proxy=true seata.config.nacos.config-type=Nacos seata.config.nacos.server-addr=127.0.0.1:8848 seata.config.nacos.namespace= seata.config.nacos.groupId=SEATA_GROUP seata.config.nacos.dataId=seata.yml seata.tm.type=xa ```
评论 19
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小钟要学习!!!

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值