MySQL分布式事务xa的介绍与使用小结

 更新时间:2024年07月21日 09:52:28   作者:雨下的竹子  
xa指的是分布式事务,传统的事务针对的是单机MySQL,本文主要介绍了MySQL分布式事务xa的介绍与使用小结,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助

1.什么是xa

xa指的是分布式事务,传统的事务针对的是单机MySQL。在如今大数据时代下,往往需要多台MySQL组成一个集群存储数据。这种情况下,针对所有MySQL节点操作就无法保证所有MySQL的事务一致性,即只要一台MySQL上执行的事务回滚了,那么其他所有MySQL也要回滚。

2.xa的组成

xa包含两个重要的角色:事务管理器(Transaction Manager)和资源管理器(Resource Manager)

  • 事务管理器:作为协调者,与针对单机的事务们进行通信,以让他们能够保持全局事务性。事务管理器通常由客户端程序实现,如后端通过调用JDBC的api来实现事务管理器,以处理分布式事务
  • 资源管理器:就是指的MySQL服务,能够支持事务

3.xa工作流程

xa在执行全局事务的过程中使用两阶段提交:

  • 准备阶段:此时资源管理器已经执行完sql操作但还没有提交,事务管理器通知资源管理器准备提交
  • 提交/回滚阶段:事务管理器通知资源管理器是提交还是回滚,只要有任意一个资源管理器出现异常,则所有资源管理器都需要回滚

4.xa语法

4.1基本命令

#1.开启一个xa事务,后面跟的字符串是xid,它是一个唯一的xa事务标识符
xa start '123';
#2.开启完xa事务后,就能执行业务sql了
update test_table set user = 'zhuzi' where id = 1;
#3.end代表业务sql执行完毕,后续可以开始二阶段提交工作了
xa end '123';
#4.第一阶段,进行准备工作
xa prepare '123';
#5.返回该MySQL上那些处于prepare阶段的xa事务的信息
xa recover;
#6.第二阶段,回滚或提交
xa commit '123';
#xa rollback '123';

注意,这些命令如果只是在一台mysql上执行是没什么意义的,就跟普通事务一样。需要客户端程序和多台mysql配合使用,才能实现分布式事务。

4.2执行xa recover时报错

ERROR 1401 (XAE03): XAER_RMERR: Fatal error occurred in the transaction branch - check your data for consistency

用户没有权限,授予权限即可

GRANT xa_recover_admin ON *.* TO root@'%';

4.3xid的组成

xid由三部分组成

  • gtrid:必须写,全局事务标识符
  • bqual:可选,分支事务的标识符(一个xa事务包含多个分支事务),默认为长度为0的空字符串""
  • formatID:可选,用于标识gtrid和bqual的格式,默认为0

程序会将这三部分取16进制拼接起来组成xid。

以上内容只是xid的一个规范,实际使用时xid只要不与其他xa事务的xid冲突即可

5.java借助jdbc使用xa事务

需要先引入jdbc的依赖,maven坐标如下:

<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    <version>8.0.28</version>
</dependency>
private static void testXA() {
    Connection connection1 = null;
    Connection connection2 = null;
    MysqlXAConnection xa1 = null;
    MysqlXAConnection xa2 = null;
    Statement statement1 = null;
    Statement statement2 = null;
    try {
        //获取两台MySQL的连接
        connection1 = DriverManager.getConnection("jdbc:mysql://192.168.86.111:3306/db3", "root", "123456");
        connection2 = DriverManager.getConnection("jdbc:mysql://192.168.86.112:3306/db1", "root", "123456");
        //获取XA连接 true表示打印日志
        xa1 = new MysqlXAConnection((JdbcConnection) connection1, true);
        xa2 = new MysqlXAConnection((JdbcConnection) connection2, true);
        //获取事务管理器
        XAResource resourceManager1 = xa1.getXAResource();
        XAResource resourceManager2 = xa2.getXAResource();

        //构造组成xid的三部分
        byte[] globalTransactionId = UUID.randomUUID().toString().replace("-", "").getBytes();
        byte[] branch1 = "分支1".getBytes();
        byte[] branch2 = "分支2".getBytes();
        int formatId = 1;
		//获取xid对象
        MysqlXid xid1 = new MysqlXid(globalTransactionId, branch1, formatId);
        MysqlXid xid2 = new MysqlXid(globalTransactionId, branch2, formatId);
        //要执行的业务sql
        String sql1 = "insert into test(`name`, `age`) values('竹子', 23)";
        String sql2 = "insert into employee(`employee_name`, `department_id`) values('竹叶', 1)";
        //执行xa start xid命令
        resourceManager1.start(xid1, XAResource.TMNOFLAGS);
        resourceManager2.start(xid2, XAResource.TMNOFLAGS);
        //执行业务sql
        statement1 = connection1.createStatement();
        statement1.execute(sql1);
        statement2 = connection2.createStatement();
        statement2.execute(sql2);
        //执行xa end xid命令
        resourceManager1.end(xid1, XAResource.TMSUCCESS);
        resourceManager2.end(xid2, XAResource.TMSUCCESS);

        //执行xa prepare xid命令
        int prepare1 = resourceManager1.prepare(xid1);
        int prepare2 = resourceManager2.prepare(xid2);
        //是否只存在一台MySQL,如果只存在一台MySQL,那么就不需要进行分布式的二阶段提交了
        boolean onePhase = false;
        //都准备好了
        if (prepare1 == XAResource.XA_OK && prepare2 == XAResource.XA_OK) {
            //执行xa commit xid命令
            resourceManager1.commit(xid1, onePhase);
            resourceManager2.commit(xid2, onePhase);
        } else {
            //执行xa rollback xid命令
            resourceManager1.rollback(xid1);
            resourceManager2.rollback(xid2);
        }
    } catch (SQLException | XAException e) {
        e.printStackTrace();
    } finally {
        try {
            statement1.close();
            statement2.close();
            xa1.close();
            xa2.close();
            connection1.close();
            connection2.close();
        } catch (SQLException throwables) {
            throwables.printStackTrace();
        }
    }
}

执行的日志信息如下:

在这里插入图片描述

到此这篇关于MySQL分布式事务xa的介绍与使用小结的文章就介绍到这了,更多相关MySQL分布式事务xa内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家! 

相关文章

  • MySQL外键级联的实现

    MySQL外键级联的实现

    本文主要介绍了MySQL外键级联的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • Mysql外键设置中的CASCADE、NO ACTION、RESTRICT、SET NULL

    Mysql外键设置中的CASCADE、NO ACTION、RESTRICT、SET NULL

    本文主要介绍了Mysql外键设置中的CASCADE、NO ACTION、RESTRICT、SET NULL,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • MySQL使用表锁和行锁的场景详解

    MySQL使用表锁和行锁的场景详解

    MySQL Innodb 的锁可以说是执行引擎的并发基础了,有了锁才能保证数据的一致性。但你知道什么时候会用表锁,什么时候会用行锁吗?本文就来和大家一起详细聊聊
    2022-09-09
  • MySQL自定义序列数的实现方式

    MySQL自定义序列数的实现方式

    这篇文章主要介绍了MySQL自定义序列数的实现方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-12-12
  • 年底了,你的mysql密码安全吗

    年底了,你的mysql密码安全吗

    这篇文章主要介绍了MySQL密码安全的相关知识,帮助大家更好的使用数据库,感兴趣的朋友可以了解下
    2021-01-01
  • Oracle和MySQL中生成32位uuid的方法举例(国产达梦同Oracle)

    Oracle和MySQL中生成32位uuid的方法举例(国产达梦同Oracle)

    近日遇到朋友问及如何生成UUID,UUID是通用唯一识别码(Universally Unique Identifier)方法,这里给大家总结下,这篇文章主要给大家介绍了关于Oracle和MySQL中生成32位uuid的方法,需要的朋友可以参考下
    2023-08-08
  • MySQL字符之char、varchar类型简析

    MySQL字符之char、varchar类型简析

    varchar和char是MySQL中的两种数据类型,都是用来存储字符串的,下面这篇文章主要给大家介绍了关于MySQL字符之char、varchar类型的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-09-09
  • mysql非主键自增长用法实例分析

    mysql非主键自增长用法实例分析

    这篇文章主要介绍了mysql非主键自增长用法,结合实例形式分析了MySQL非主键自增长的基本设置、使用方法与操作注意事项,需要的朋友可以参考下
    2020-02-02
  • k8s搭建mysql集群实现主从复制的方法步骤

    k8s搭建mysql集群实现主从复制的方法步骤

    本文是基于已有k8s环境下,介绍在k8s环境中部署mysql主从集群的实现步骤,对mysql学习有一定的帮助,感兴趣的可以学习一下
    2023-01-01
  • Windows XP系统安装MySQL5.5.28图解教程

    Windows XP系统安装MySQL5.5.28图解教程

    很多朋友在winxp系统中开发php等,需要安装mysql数据库,这里简单介绍下,如何在xp下安装mysql软件,其实跟其它系统都差不多,主要是软件对系统的兼容性
    2013-05-05

最新评论