Seata AT 模式 原理详解

目录

前言

seata框架简介

AT 前提

AT 整体机制

写隔离

读隔离

工作机制

一阶段

二阶段-回滚

二阶段-提交

附录

回滚日志表


前言

seata框架简介

在 Seata 的架构中,一共有三个角色:

  • TC (Transaction Coordinator) - 事务协调者:维护全局和分支事务的状态,驱动全局事务提交或回滚。
  • TM (Transaction Manager) - 事务管理器:定义全局事务的范围,开始全局事务、提交或回滚全局事务。
  • RM ( Resource Manager ) - 资源管理器:管理分支事务处理的资源( Resource ),与 TC 交谈以注册分支事务和报告分支事务的状态,并驱动分支事务提交或回滚。

其中,TC 为单独部署的 Server 服务端,TM 和 RM 为嵌入到应用中的 Client 客户端

在 Seata 中,一个分布式事务的生命周期如下:

  • TM 请求 TC 开启一个全局事务。TC 会生成一个 XID 作为该全局事务的编号。

    XID,会在微服务的调用链路中传播,保证将多个微服务的子事务关联在一起。

  • RM 请求 TC 将本地事务注册为全局事务的分支事务,通过全局事务的 XID 进行关联。

  • TM 请求 TC 告诉 XID 对应的全局事务是进行提交还是回滚。

  • TC 驱动 RM 们将 XID 对应的自己的本地事务进行提交还是回滚。

AT 前提

  • 基于支持本地 ACID 事务的关系型数据库。
  • Java 应用,通过 JDBC 访问数据库。

AT 整体机制

两阶段提交协议的演变:

  • 一阶段:业务数据和回滚日志记录在同一个本地事务中提交,释放本地锁和连接资源。

  • 二阶段:

    • 提交异步化,非常快速地完成。
    • 回滚通过一阶段的回滚日志进行反向补偿。

写隔离

  • 一阶段本地事务提交前,需要确保先拿到 全局锁 。
  • 拿不到 全局锁 ,不能提交本地事务。
  • 拿 全局锁 的尝试被限制在一定范围内,超出范围将放弃,并回滚本地事务,释放本地锁。

以一个示例来说明:

两个全局事务 tx1 和 tx2,分别对 a 表的 m 字段进行更新操作,m 的初始值 1000。

tx1 先开始,开启本地事务,拿到本地锁,更新操作 m = 1000 - 100 = 900。本地事务提交前,先拿到该记录的 全局锁 ,本地提交释放本地锁。 tx2 后开始,开启本地事务,拿到本地锁,更新操作 m = 900 - 100 = 800。本地事务提交前,尝试拿该记录的 全局锁 ,tx1 全局提交前,该记录的全局锁被 tx1 持有,tx2 需要重试等待 全局锁 。

 

tx1 二阶段全局提交,释放 全局锁 。tx2 拿到 全局锁 提交本地事务。

 

如果 tx1 的二阶段全局回滚,则 tx1 需要重新获取该数据的本地锁,进行反向补偿的更新操作,实现分支的回滚。

此时,如果 tx2 仍在等待该数据的 全局锁,同时持有本地锁,则 tx1 的分支回滚会失败。分支的回滚会一直重试,直到 tx2 的 全局锁 等锁超时,放弃 全局锁 并回滚本地事务释放本地锁,tx1 的分支回滚最终成功。

因为整个过程 全局锁 在 tx1 结束前一直是被 tx1 持有的,所以不会发生 脏写 的问题。

读隔离

在数据库本地事务隔离级别 读已提交(Read Committed) 或以上的基础上,Seata(AT 模式)的默认全局隔离级别是 读未提交(Read Uncommitted) 。

如果应用在特定场景下,必需要求全局的 读已提交 ,目前 Seata 的方式是通过 SELECT FOR UPDATE 语句的代理。

 

SELECT FOR UPDATE 语句的执行会申请 全局锁 ,如果 全局锁 被其他事务持有,则释放本地锁(回滚 SELECT FOR UPDATE 语句的本地执行)并重试。这个过程中,查询是被 block 住的,直到 全局锁 拿到,即读取的相关数据是 已提交 的,才返回。

出于总体性能上的考虑,Seata 目前的方案并没有对所有 SELECT 语句都进行代理,仅针对 FOR UPDATE 的 SELECT 语句。

工作机制

以一个示例来说明整个 AT 分支的工作过程。

业务表:product

FieldTypeKey
idbigint(20)PRI
namevarchar(100) 
sincevarchar(100) 

AT 分支事务的业务逻辑:

update product set name = 'GTS' where name = 'TXC';

一阶段

过程:

  1. 解析 SQL:得到 SQL 的类型(UPDATE),表(product),条件(where name = 'TXC')等相关的信息。
  2. 查询前镜像:根据解析得到的条件信息,生成查询语句,定位数据。
select id, name, since from product where name = 'TXC';

得到前镜像:

idnamesince
1TXC2014
  1. 执行业务 SQL:更新这条记录的 name 为 'GTS'。
  2. 查询后镜像:根据前镜像的结果,通过 主键 定位数据。
select id, name, since from product where id = 1`;

得到后镜像:

idnamesince
1GTS2014
  1. 插入回滚日志:把前后镜像数据以及业务 SQL 相关的信息组成一条回滚日志记录,插入到 UNDO_LOG 表中。
{
	"branchId": 641789253,
	"undoItems": [{
		"afterImage": {
			"rows": [{
				"fields": [{
					"name": "id",
					"type": 4,
					"value": 1
				}, {
					"name": "name",
					"type": 12,
					"value": "GTS"
				}, {
					"name": "since",
					"type": 12,
					"value": "2014"
				}]
			}],
			"tableName": "product"
		},
		"beforeImage": {
			"rows": [{
				"fields": [{
					"name": "id",
					"type": 4,
					"value": 1
				}, {
					"name": "name",
					"type": 12,
					"value": "TXC"
				}, {
					"name": "since",
					"type": 12,
					"value": "2014"
				}]
			}],
			"tableName": "product"
		},
		"sqlType": "UPDATE"
	}],
	"xid": "xid:xxx"
}
  1. 提交前,向 TC 注册分支:申请 product 表中,主键值等于 1 的记录的 全局锁 。
  2. 本地事务提交:业务数据的更新和前面步骤中生成的 UNDO LOG 一并提交。
  3. 将本地事务提交的结果上报给 TC。

二阶段-回滚

  1. 收到 TC 的分支回滚请求,开启一个本地事务,执行如下操作。
  2. 通过 XID 和 Branch ID 查找到相应的 UNDO LOG 记录。
  3. 数据校验:拿 UNDO LOG 中的后镜与当前数据进行比较,如果有不同,说明数据被当前全局事务之外的动作做了修改。这种情况,需要根据配置策略来做处理,详细的说明在另外的文档中介绍。
  4. 根据 UNDO LOG 中的前镜像和业务 SQL 的相关信息生成并执行回滚的语句:
update product set name = 'TXC' where id = 1;

     5. 提交本地事务。并把本地事务的执行结果(即分支事务回滚的结果)上报给 TC。

二阶段-提交

  1. 收到 TC 的分支提交请求,把请求放入一个异步任务的队列中,马上返回提交成功的结果给 TC。
  2. 异步任务阶段的分支提交请求将异步和批量地删除相应 UNDO LOG 记录。

附录

回滚日志表

UNDO_LOG Table:不同数据库在类型上会略有差别。

以 MySQL 为例:

FieldType
branch_idbigint PK
xidvarchar(100)
contextvarchar(128)
rollback_infolongblob
log_statustinyint
log_createddatetime
log_modifieddatetime
-- 注意此处0.7.0+ 增加字段 context
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,
  PRIMARY KEY (`id`),
  UNIQUE KEY `ux_undo_log` (`xid`,`branch_id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

转自:https://seata.io/zh-cn/docs/dev/mode/at-mode.html

史上最强Tomcat8性能优化

阿里巴巴为什么能抗住90秒100亿?--服务端高并发分布式架构演进之路

B2B电商平台--ChinaPay银联电子支付功能

学会Zookeeper分布式锁,让面试官对你刮目相看

SpringCloud电商秒杀微服务-Redisson分布式锁方案

查看更多好文,进入公众号--撩我--往期精彩

一只 有深度 有灵魂 的公众号0.0

<think>嗯,用户现在想了解如何部署SeataAT模式,询问具体的部署步骤和配置方法。我之前回答过关于Seata的TCC和Saga模式是否需要接口的问题,现在需要根据用户提供的引用和之前的知识来组织回答。 首先,用户提供的引用1提到了搭建Spring Boot集成Seata的示例,以及前置条件中的Seata Server安装。引用2是关于使用nacos-config.sh脚本配置Nacos的步骤,引用3则涉及AT模式原理,尤其是二阶段提交和UNDO LOG的处理。需要整合这些信息,但要注意用户现在关心的是部署步骤,而不是原理。 接下来,我需要回忆Seata AT模式的部署流程。通常部署Seata需要安装Seata Server,配置注册中心和配置中心(比如Nacos),然后在客户端应用中添加依赖和配置。引用2中提到的nacos-config.sh脚本是用来将Seata的配置推送到Nacos中的,这应该是配置中心的一部分。引用1提到避免脏写和脏读,可能需要在配置时注意数据源代理和UNDO_LOG表的创建。 用户可能希望步骤清晰,分阶段说明,比如前置条件、服务端部署、客户端集成、注意事项等。需要确保每一步都有具体的操作指导,比如如何执行脚本,如何修改配置文件等。同时,注意事项部分可以包括脏写脏读的避免,配置检查等,参考引用1和3的内容。 还要注意用户可能对配置文件的路径、脚本执行参数等细节有疑问,需要详细说明,比如在引用2中提到的执行nacos-config.sh时的参数示例,应该给出具体的命令示例,如sh nacos-config.sh -h Nacos地址 -t token等。此外,客户端配置部分需要说明application.yml中的必要配置项,如事务组名称、注册中心类型、配置中心设置等。 另外,用户之前的问题是关于TCC和Saga是否需要接口,而这次是关于AT模式的部署,可能需要区分不同模式的不同配置,但用户现在只关注AT,所以重点放在AT的部署步骤上。 最后,生成的相关问题需要与部署和配置相关,比如如何选择注册中心,如何调试事务,生产环境注意事项等,帮助用户进一步了解后续可能遇到的问题。</think>### **Seata AT模式部署步骤及配置详解** --- #### **一、部署前提** 1. **环境要求**: - JDK 1.8+ - MySQL 5.7+(存储事务日志与业务数据) - Nacos(或其他注册中心,如Eureka、Zookeeper) - 已下载Seata Server(如`seata-server-1.4.2.zip`)[^1] --- #### **二、Seata Server部署** ##### **1. 启动Seata Server** - **步骤**: 1. 解压Seata Server包,进入`conf`目录。 2. 修改`registry.conf`,配置注册中心(以Nacos为例): ```properties registry { type = "nacos" nacos { application = "seata-server" serverAddr = "127.0.0.1:8848" # Nacos地址 namespace = "" cluster = "default" } } ``` 3. 启动Seata Server: ```bash sh ./bin/seata-server.sh -p 8091 -h 0.0.0.0 -m db # 使用数据库模式存储事务日志[^1] ``` ##### **2. 导入Nacos配置** - **步骤**: 1. 进入Seata的`script/config-center/nacos`目录,找到`nacos-config.sh`。 2. 执行脚本,将Seata配置推送至Nacos: ```bash sh nacos-config.sh -h 127.0.0.1 -p 8848 -t <Nacos命名空间Token> -g SEATA_GROUP[^2] ``` 3. 验证Nacos控制台是否生成`seata.properties`配置项[^2]。 --- #### **三、客户端(业务服务)配置** ##### **1. 添加依赖** - **Maven依赖**: ```xml <dependency> <groupId>io.seata</groupId> <artifactId>seata-spring-boot-starter</artifactId> <version>1.4.2</version> </dependency> ``` ##### **2. 配置数据源代理** - **修改`application.yml`**: ```yaml seata: enabled: true tx-service-group: my_tx_group # 事务组名称,需与Seata Server配置一致 service: vgroup-mapping: my_tx_group: default # 事务组映射到Seata集群 registry: type: nacos nacos: server-addr: 127.0.0.1:8848 config: type: nacos nacos: server-addr: 127.0.0.1:8848 ``` ##### **3. 初始化数据库** - **创建`UNDO_LOG`表**(用于AT模式的事务回滚): ```sql CREATE TABLE IF NOT EXISTS `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, PRIMARY KEY (`id`), UNIQUE KEY `ux_undo_log` (`xid`, `branch_id`) ) ENGINE = InnoDB AUTO_INCREMENT = 1 DEFAULT CHARSET = utf8; ``` --- #### **四、验证部署** 1. **启动业务服务**:确保服务注册到Nacos并连接到Seata Server。 2. **测试分布式事务**: - 调用涉及多个微服务的业务接口,观察事务是否正常提交或回滚。 - 检查`UNDO_LOG`表是否生成回滚日志[^3]。 --- #### **五、注意事项** 1. **避免脏写与脏读**: - 确保所有参与事务的数据库表均有主键。 - 业务SQL需支持Seata的全局锁机制(如避免跨服务直接更新同一行数据)[^1][^3]。 2. **配置一致性**: - 事务组名称(`tx-service-group`)需在客户端和Seata Server中保持一致。 3. **高可用部署**: - Seata Server建议集群部署,并通过Nacos集群提升可用性[^1]。 --- ### **相关问题** 1. **Seata AT模式如何选择注册中心(Nacos vs Eureka)?** Nacos支持动态配置推送,适合生产环境;Eureka需手动同步配置,适合简单场景[^2]。 2. **如何调试Seata AT模式的事务回滚失败问题?** 检查`UNDO_LOG`表记录、业务SQL是否触发全局锁冲突,或开启Seata的debug日志[^3]。 3. **Seata AT模式在生产环境中需注意哪些性能瓶颈?** 高频事务场景需优化数据库连接池、异步化二阶段提交,并监控全局锁竞争[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值