一个注解搞定分布式事务

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
云数据库 RDS MySQL,高可用系列 2核4GB
简介: 一个注解搞定分布式事务

大家好,我是不才陈某~

陈某的《Spring Cloud Alibaba实战项目》 视频教程已经录完了,涉及到Alibaba的各种中间件、OAuth2微服务认证鉴权全链路灰度发布分布式事务实战,戳这里--->Spring Cloud Alibaba 实战 视频专栏 开放订阅~

这是《亿级数据分库分表实战》专栏的第6篇文章,往期文章如下:

上篇介绍了Sharding-JDBC的本地事务在网络、硬件故障下是无法保证分布式事务。

Sharding-JDBC目前还整合其他分布式事务模式来解决事务问题,今天就来介绍一下两阶段事务XA是如何解决分布式事务

什么是XA规范

XA 规范 是 X/Open 组织定义的分布式事务处理(DTP,Distributed Transaction Processing)标准。

XA 规范描述了全局的事务管理器与局部的资源管理器之间的接口。XA规范的目的是允许多个资源(如数据库,应用服务器,消息队列等)在同一事务中访问,这样可以使 ACID 属性跨越应用程序而保持有效。

XA 规范使用两阶段提交来保证所有资源同时提交或回滚任何特定的事务。

关于两阶段提交事务模式不清楚的可以看之前的文章:七种分布式事务解决方案

XA 事务的基础是两阶段提交协议。需要有一个事务协调者来保证所有的事务参与者都完成了准备工作(第一阶段)。如果协调者收到所有参与者都准备好的消息,就会通知所有的事务都可以提交了(第二阶段)。

MySQL 在这个 XA 事务中扮演的是参与者的角色,而不是协调者(事务管理器)。

MySQL中的XA事务分为外部XA和内部XA:

  1. 外部XA:可以参与外部的分布式事务,需要协调者参与协调
  2. 内部XA:用于同一实例下跨多引擎事务,由 Binlog 作为协调者,比如在一个存储引擎提交时,需要将提交信息写入二进制日志,这就是一个分布式内部 XA 事务,只不过二进制日志的参与者是 MySQL 本身。

基于XA规范分布式事务对业务是无侵入的,用户可以像使用本地事务一样使用基于XA规范的分布式事务。

但是基于XA规范的分布式事务属于强一致性事务,性能比较低,尤其在长事务的情况下,对资源的锁定时间较长,一般适用于短事务、低并发的场景

MySQL中XA事务

接下来就以为MySQL中XA来演示一下,如下;

上述命令什么意思呢?解析如下:

  1. XA START "test_xid":开启一个XA事务,test_xid是全局事务xid,名称任意
  2. update product_base set price=1000 where product_id=743948772064624640:执行的SQL语句
  3. XA END "test_xid":这个表示结束一个 XA 事务,此时事务的状态转为 IDLE
  4. XA PREPARE "test_xid":这个将事务置为 PREPARE 状态
  5. XA COMMIT "test_xid":这个用来提交事务,提交之后,事务的状态就是 COMMITED。
  6. XA ROLLBACK "test_xid":这个用来回滚事务

Sharding-JDBC中的XA事务

Sharding-JDBC目前已经整合了XA事务的支持,支持的Atomikos、NARAYANA 、BITRONIX 事务的实现,默认的实现是Atomikos。下面将以Atomikos为例子介绍一下Sharding-JDBC如何实现XA事务

SharingSphere是通过SPI的方式去定制分布式事务解决方案的,XA的默认实现是org.apache.shardingsphere.transaction.xa.XAShardingTransactionManager,有兴趣的可以看下其中源码

后文介绍到的弱一致性事务使用Seata的AT模式,对应的是org.apache.shardingsphere.transaction.base.seata.at.SeataATShardingTransactionManager

当然Seata也是支持XA事务,如果后续改写的话,可以通过SPI的方式的定制

1. 添加依赖

使用XA事务需要添加一个依赖,如下:

<!-- 使用XA事务时,需要引入此模块 -->
<dependency>
    <groupId>org.apache.shardingsphere</groupId>
    <artifactId>sharding-transaction-xa-core</artifactId>
</dependency>

可以看到这个依赖中内置了ATOMIKOS,如下:

2. 配置事务管理器

使用事务之前当然需要配置一个事务管理器,这里和Spring Boot 中配置没差别,如下:

@Configuration
@EnableTransactionManagement
public class TransactionConfiguration {
    @Bean
    public PlatformTransactionManager txManager(final DataSource dataSource) {
        return new DataSourceTransactionManager(dataSource);
    }
}

3. @ShardingTransactionType注解

想要开启XA分布式事务还需要使用一个注解@ShardingTransactionType,其中的value属性的值如下:

  1. TransactionType.LOCAL:开启本地事务的支持,默认值
  2. TransactionType.XA:开启XA事务的支持
  3. TransactionType.BASE:开启弱事务的支持

那么显然这里只需要配置TransactionType.XA就能开启XA事务的支持

伪代码如下:

@ShardingTransactionType(value = TransactionType.XA)
@Transactional
public void method(){
    //跨库操作
    ..............
}

4. 启动测试

演示的源码请看视频教程中的源码,需要注意的是Atomikos会在项目的logs目录下生成xa_tx.log文件,其中记录了事务日志,在XA崩溃恢复时所需要的日志,一定不要删除

对于Atomikos的一些配置可以通过在项目的classpath中添加jta.properties来定制化Atomikos配置项,这里就不再给大家详细介绍了。

相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
5月前
|
NoSQL 安全 Java
nicelock--一个注解即可使用Redis分布式锁!
Nicelock的引入为分布式系统中的资源同步访问提供了一个简单高效和可靠的解决方案。通过注解的方式,简化了锁的实现和使用,使开发人员可以将更多精力专注于业务逻辑的实现,而不是锁的管理。此外,Nicelock在保持简单易用的同时,也提供了足够的灵活性和可靠性,满足了不同应用场景下对分布式锁的需求。
73 1
|
8月前
|
缓存 NoSQL Java
【亮剑】分布式锁是保证多服务实例同步的关键机制,常用于互斥访问共享资源、控制访问顺序和系统保护,如何使用注解来实现 Redis 分布式锁的功能?
【4月更文挑战第30天】分布式锁是保证多服务实例同步的关键机制,常用于互斥访问共享资源、控制访问顺序和系统保护。基于 Redis 的分布式锁利用 SETNX 或 SET 命令实现,并考虑自动过期、可重入及原子性以确保可靠性。在 Java Spring Boot 中,可通过 `@EnableCaching`、`@Cacheable` 和 `@CacheEvict` 注解轻松实现 Redis 分布式锁功能。
141 0
|
存储 NoSQL Java
一个注解实现分布式锁
一个注解实现分布式锁
124 0
|
消息中间件 NoSQL JavaScript
Spring Boot加一个注解,轻松实现 Redis 分布式锁
Spring Boot加一个注解,轻松实现 Redis 分布式锁
35441 2
|
消息中间件 NoSQL JavaScript
使用注解实现redis分布式锁
使用注解实现redis分布式锁
|
NoSQL Java Redis
如何使用注解来实现 Redis 分布式锁的功能?
如何使用注解来实现 Redis 分布式锁的功能?
163 0
|
NoSQL Java Redis
完美,SpringBoot中使用注解来实现 Redis 分布式锁
对于耗时业务和核心数据,不能让重复的请求同时操作数据,避免数据的不正确,所以要使用分布式锁来对它们进行保护。有些业务请求,属于耗时操作,需要加锁,防止后续的并发操作,同时对数据库的数据进行操作,需要避免对之前的业务造成影响。
264 0
|
NoSQL Java Redis
如何使用注解实现分布式锁
如何使用注解实现分布式锁
335 0
|
XML NoSQL Java
【SimpleFunction系列二.2】SpringBoot注解整合Redisson分布式锁
Redisson是架设在Redis基础上的一个Java驻内存数据网格(In-Memory Data Grid)。充分的利用了Redis键值数据库提供的一系列优势,基于Java实用工具包中常用接口,为使用者提供了一系列具有分布式特性的常用工具类。
400 0
【SimpleFunction系列二.2】SpringBoot注解整合Redisson分布式锁
|
NoSQL Redis
注解的方式实现redis分布式锁
创建redisLock注解: import java.lang.annotation.Documented; import java.lang.annotation.ElementType; import java.
1213 0