|
| 1 | +--- |
| 2 | +title: Seata实战入门与实战 |
| 3 | +date: 2024-12-30 19:07:05 |
| 4 | +updated: 2024-12-30 19:07:05 |
| 5 | +tags: |
| 6 | + - 分布式 |
| 7 | + - Seata |
| 8 | +comments: true |
| 9 | +categories: |
| 10 | + - 分布式 |
| 11 | + - Seata |
| 12 | +thumbnail: https://cdn.jsdelivr.net/gh/hackerHiJu/note-picture@main/note-picture/%25E5%25A4%25A9%25E7%25A9%25BA.png |
| 13 | +--- |
| 14 | + |
| 15 | +# 分布式事务 |
| 16 | + |
| 17 | +## 一、分布式事务的组成部分 |
| 18 | + |
| 19 | +- 事务参与者:对应的一个一个的微服务 |
| 20 | +- 资源服务器:对应一个个微服务的数据库 |
| 21 | +- 事务管理器:决策各个事务参与者的提交和回滚 |
| 22 | + |
| 23 | +### 两阶段提交: |
| 24 | + |
| 25 | +1. 准备阶段:向事务管理器向事务参与者发送预备请求,事务参与者在写本地的redo和undo日志,但是不提交,并且返回准备就绪的信息,最后提交的动作交给第二阶段来进行 |
| 26 | +2. 提交阶段:如果事务协调者收到失败或者超时的信息,直接给每个参与者发送回滚消息;否则提交消息,最后根据协调者的指令释放所有事务处理过程中使用的资源锁 |
| 27 | + |
| 28 | + |
| 29 | + |
| 30 | +## 二、项目例子 |
| 31 | + |
| 32 | +当前依赖,全局事务XID,不需要手动进行绑定,自动进行传递 |
| 33 | + |
| 34 | +```java |
| 35 | +<dependency> |
| 36 | + <groupId>com.alibaba.cloud</groupId> |
| 37 | + <!–加入spring-cloud-alibaba-seata,解决xid不传递问题–> |
| 38 | + <artifactId>spring-cloud-alibaba-seata</artifactId> |
| 39 | + <version>2.2.0.RELEASE</version> |
| 40 | + <exclusions> |
| 41 | + <exclusion> |
| 42 | + <groupId>io.seata</groupId> |
| 43 | + <artifactId>seata-spring-boot-starter</artifactId> |
| 44 | + </exclusion> |
| 45 | + <exclusion> |
| 46 | + <groupId>io.seata</groupId> |
| 47 | + <artifactId>seata-all</artifactId> |
| 48 | + </exclusion> |
| 49 | + </exclusions> |
| 50 | +</dependency> |
| 51 | + |
| 52 | +``` |
| 53 | + |
| 54 | +全局事务XID需要通过过滤器或拦截器进行手动绑定,否则下游服务获取不到全局XID回滚不了 |
| 55 | + |
| 56 | +```java |
| 57 | +<dependency> |
| 58 | + <groupId>io.seata</groupId> |
| 59 | + <artifactId>seata-spring-boot-starter</artifactId> |
| 60 | + <version>1.3.0</version> |
| 61 | + <!-- 这里需要排除自身的seata-all --> |
| 62 | + <exclusions> |
| 63 | + <exclusion> |
| 64 | + <artifactId>seata-all</artifactId> |
| 65 | + <groupId>io.seata</groupId> |
| 66 | + </exclusion> |
| 67 | + </exclusions> |
| 68 | + </dependency> |
| 69 | + <!-- 导入与之前下载的seata版本一致的包 --> |
| 70 | + <dependency> |
| 71 | + <groupId>io.seata</groupId> |
| 72 | + <artifactId>seata-all</artifactId> |
| 73 | + <version>1.3.0</version> |
| 74 | + </dependency> |
| 75 | +``` |
| 76 | + |
| 77 | +OpenFeign进行手动传递XID |
| 78 | + |
| 79 | +```java |
| 80 | +@Component |
| 81 | +public class FeignConfiguration implements RequestInterceptor { |
| 82 | + |
| 83 | + @Override |
| 84 | + public void apply(RequestTemplate requestTemplate) { |
| 85 | + requestTemplate.header("XID", RootContext.getXID()); |
| 86 | + } |
| 87 | +} |
| 88 | +``` |
| 89 | + |
| 90 | +提供者手动绑定XID |
| 91 | + |
| 92 | +```java |
| 93 | +@Component |
| 94 | +public class SeataFilter implements Filter { |
| 95 | + @Override |
| 96 | + public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException { |
| 97 | + HttpServletRequest request = (HttpServletRequest) servletRequest; |
| 98 | + //手动绑定XID |
| 99 | + String xid = request.getHeader("XID"); |
| 100 | + if(StringUtils.isNotBlank(xid)){ |
| 101 | + RootContext.bind(xid); |
| 102 | + } |
| 103 | + filterChain.doFilter(servletRequest,servletResponse); |
| 104 | + } |
| 105 | +} |
| 106 | +``` |
| 107 | + |
| 108 | +**seata在1.0.0版本之后就不需要手动进行数据源代理,已经被自动代理** |
| 109 | + |
| 110 | +客户端的配置文件 |
| 111 | + |
| 112 | +```yml |
| 113 | +seata: |
| 114 | + enabled: true |
| 115 | + tx-service-group: my_first_seata #配置文件中的事务服务组一样 |
| 116 | + config: |
| 117 | + type: nacos # nacos中拉去对应的配置文件 |
| 118 | + nacos: |
| 119 | + server-addr: 192.168.60.46:8849 |
| 120 | + group: SEATA_GROUP |
| 121 | + registry: # 会去nacos中拉去seata-server服务 |
| 122 | + type: nacos |
| 123 | + nacos: |
| 124 | + application: seata-server |
| 125 | + server-addr: 192.168.60.46:8849 |
| 126 | + group: SEATA_GROUP |
| 127 | +``` |
| 128 | +
|
| 129 | +**seata1.0.0之后config文件下就移除了nacos-config.txt等文件,改为了config.txt需要手动下载,并且config.txt需要在nacos-config.sh的上一级目录下才能推送到nacos中** |
| 130 | +
|
| 131 | +```application |
| 132 | +# 只需要修改下面几种配置即可,这里是配置客户端需要拉取的配置文件 |
| 133 | +service.vgroupMapping.自定义的名称=default |
| 134 | +store.mode=db #修改为db |
| 135 | +store.db.dbType=mysql #修改msql的连接方式账号和密码 |
| 136 | +``` |
| 137 | + |
| 138 | + |
| 139 | + |
| 140 | +## 三、seata原理 |
| 141 | + |
| 142 | +### 1、角色划分 |
| 143 | + |
| 144 | +- RM:资源管理者/事务参与者,也可以是一个TM |
| 145 | + |
| 146 | +- TM:事务管理者,也是一个微服务,充当分布式事务的发起者 |
| 147 | + |
| 148 | +- TC:全局事务协调者seata-server,一个包需要搭建,TC来决定事务的回滚和提交 |
| 149 | + |
| 150 | +### 2、AT模式 |
| 151 | + |
| 152 | +#### (1)核心概念 |
| 153 | + |
| 154 | +- 两阶段提交:只执行,不提交 |
| 155 | + |
| 156 | +- seata 核心概念:边执行,边提交(两阶段的变种) |
| 157 | + - 一阶段:查询前置快照---------->执行业务语句-------------->查询出后置快照,保存只undo_log日志表中 |
| 158 | + - 二阶段提交:分支插入待删除队列--------->异步删除undo_log表中数据 |
| 159 | + - 二阶段回滚:根据配置选项选择是否检验dirty data------------>构造方向SQL----------->删除undo_log |
| 160 | + |
| 161 | +#### (2)执行流程 |
| 162 | + |
| 163 | +阶段一:业务SQL:update product set name = 'GTS' where name = 'TXC' |
| 164 | + |
| 165 | +- 解析SQL,根据update product解析出update语句,表product,条件where等相关信息 |
| 166 | +- 查询前置镜像:根据解析sql生成查询语句:**select id**, **name**, since **from** product **where name** = 'TXC' |
| 167 | +- 执行业务SQL:update product set name = 'GTS' where name = 'TXC' 更新数据 |
| 168 | +- 查询后置镜像:通过主键定位数据 |
| 169 | +- 插入回滚日志:把前后镜像数据以及业务SQL相关的信息组成一条回滚日志记录,插入到undo_log表中 |
| 170 | +- 提交前,向TC注册分支,申请product表中,主键值记录的全局锁 |
| 171 | +- 本地事务提交:业务数据的更新和前面步骤中生成的undo_log一并提交 |
| 172 | +- 将本地事务的提交结果上报给TC |
| 173 | + |
| 174 | +**业务数据和回滚日志记录会在同一个本地事务中保存,会释放本地锁和连接资源** |
| 175 | + |
| 176 | +阶段二(回滚): |
| 177 | + |
| 178 | +- 收到TC的分支回滚请求,开启一个本地事务,把请求放入一个异步任务的队列里面 |
| 179 | +- 根据XID和Branch ID查找到相应的undo_log记录 |
| 180 | +- 数据校验:拿undo_log中的后镜与当前数据进行比较,如果有不同,说明当前数据被其它事务所更改,需要通过配置的策略进行处理 |
| 181 | +- 根据undo_log的前置镜像和业务sql的相关信息组成回滚语句 |
| 182 | +- 将分支回滚的结果提交给TC |
| 183 | + |
| 184 | +通过一阶段的回滚日志进行反向补偿 |
| 185 | + |
| 186 | +阶段二(提交): |
| 187 | + |
| 188 | +- 收到TC的分支提交请求,把请求放入异步队列中,马上返回提交成功的结果给TC |
| 189 | +- 异步批量的删除undo_log记录 |
| 190 | + |
| 191 | +#### (3)写隔离 |
| 192 | + |
| 193 | +**一阶段提交本地事务,必须需要拿到更改数据的全局锁,拿不到全局锁,不能提交本地事务,超出等待时间,会回滚本地事务,释放本地锁** |
| 194 | + |
| 195 | +例:tx1和tx2两个全局事务同时修改 a表的m字段,m初始为1000; |
| 196 | + |
| 197 | +tx1先开始,拿到本地锁,将m 1000-100 = 900。本地事务提交前,先拿到该记录的全局锁,本地提交释放本地锁。tx2开始,拿到本地锁,将m 900-100=800,提交本地事务前,先获取该记录的全局锁,tx1全局事务提交前,全局锁会被tx1所持有,tx2就会重试等待全局锁。 |
| 198 | + |
| 199 | +**tx1二阶段全局提交,释放全局锁。tx2拿到全局锁提交本地事务。如果tx1二阶段为全局回滚,那么会重新重试获取本地锁,此时tx2如果还在等待全局锁,同时持有本地锁,tx1分支事务就会等待tx2超时释放本地锁之后,再次获取本地锁;整个过程 全局锁都是被 tx1锁持有,不会存在脏数据的问题** |
| 200 | + |
| 201 | +#### (4)读隔离 |
| 202 | + |
| 203 | +Seata AT模式的默认全局隔离级别是读未提交,如果在特定场景下,必需要求全局的读已提交,Seata采用通过select for update 语句来进行代理的;select for update语句的执行会申请*全局锁* ,如果全局锁被其它事务锁持有,就会回滚select for update的本地执行并且重试,因为这时候查询是被锁住,直到全局锁拿到,即读取相关的数据是已提交的 |
| 204 | + |
| 205 | +### 3、TCC模式 |
| 206 | + |
| 207 | +AT模式是基于本地支持ACID事务的关系型数据库: |
| 208 | + |
| 209 | +- 一阶段prepare行为:在本地事务中,一并提交数据更新和相应的回滚记录 |
| 210 | +- 二阶段commit行为:马上成功,自动异步删除回滚记录 |
| 211 | +- 二阶段rollback行为:通过回滚日志,自动生成补偿操作,完成数据回滚 |
| 212 | + |
| 213 | +相应的TCC模式,不依赖本地底层数据的事务支持: |
| 214 | + |
| 215 | +- 一阶段prepare行为:调用自定义的prepare逻辑 |
| 216 | +- 二阶段commit行为:调用自定义的commit逻辑 |
| 217 | +- 二阶段rollback行为:调用自定义的rollback逻辑 |
| 218 | + |
| 219 | +### 4、Saga模式 |
| 220 | + |
| 221 | +- 特点:业务流程中每个参与者都提交本地事务,当某一个参与者失败则补偿前面已经成功的参与者,一阶段正向服务和二阶段补偿服务都由业务开发者实现 |
| 222 | +- sage实现:基于状态机引擎来实现 |
| 223 | + - 通过状态图来定义服务调用的流程并生成json状态语言定义文件 |
| 224 | + - 状态图中一个节点可以是调用一个服务,节点可以配置它的补偿节点 |
| 225 | + - 状态图json由状态机引擎驱动执行,当出现异常时状态引擎反向执行已经成功节点对应的补偿节点将事务回滚(用户可以自定义是否进行补偿) |
| 226 | + - 可以实现服务编排需求,支持单项选择、并发、子流程、参数转换、参数映射、服务执行状态判断、异常捕获等功能 |
| 227 | + |
| 228 | +### 5、XA模式 |
| 229 | + |
| 230 | +- 特点:利用事务资源(数据库、消息服务等)对 XA 协议的支持,以 XA 协议的机制来管理分支事务的一种 事务模式 |
| 231 | + |
0 commit comments