【分布式事务AT模式 SpringCloud集成Seata框架】分布式事务框架Seata详细讲解

news2025/1/23 17:46:36

前言

上篇文章我们讲述了如何启动seata的本地服务,并且注册到nacos使用,这篇文章将在SpringCloud中整合Seata框架

上篇文章传送门:https://blog.csdn.net/Syals/article/details/130102851?spm=1001.2014.3001.5501

本篇主要内容:@GlobalTransactional注解

  • @GlobalTransactional是Seata框架提供的注解,用于开启一个全局事务。当一个方法被标记为@GlobalTransactional时,Seata框架会自动创建一个全局事务,并将该方法的执行视为整个事务的一个参与者。

  • 在使用@GlobalTransactional注解时,需要在Seata Server中配置好相应的事务组,并使用相同的事务组ID和事务模式。同时,所有参与该全局事务的服务都需要使用相同的事务组ID和事务模式,并在业务代码中使用@Transactional注解开启本地事务。

  • 当全局事务中的任何一个本地事务发生异常时,Seata框架会回滚整个全局事务,保证数据的一致性。同时,Seata框架还提供了一系列的扩展点和机制,可以自定义全局事务的创建、提交和回滚过程,以满足不同场景的需求。

  • 使用@GlobalTransactional注解可以简化分布式事务的管理和操作,提高开发效率和数据一致性。

版本

版本号
jdk1.8
SpringBoot2.3.12.RELEASE
SpringCloud2.2.7.RELEASE
SpringCloudVersionHoxton.SR12

准备工作

  • 创建一个新的数据库seata-test,创建数据库表it_orderit_stockundo_log
    其中it_order是我们的订单表,it_stock为库存表,undo_log则为seata框架要求的事务日志表,在你每个要操作事务的数据库中都要有一个undo_log表

  • 这里的话数据表随便也可以,我现在就按演示的数据表来进行操作,但是undo_log表是固定的

undo_log表:

-- ----------------------------
-- Table structure for undo_log
-- ----------------------------
DROP TABLE IF EXISTS `undo_log`;
CREATE TABLE `undo_log`  (
  `id` bigint(0) NOT NULL AUTO_INCREMENT,
  `branch_id` bigint(0) NOT NULL,
  `xid` varchar(100) CHARACTER SET utf8mb3 COLLATE utf8mb3_general_ci NOT NULL,
  `context` varchar(128) CHARACTER SET utf8mb3 COLLATE utf8mb3_general_ci NOT NULL,
  `rollback_info` longblob NOT NULL,
  `log_status` int(0) NOT NULL,
  `log_created` datetime(0) NOT NULL,
  `log_modified` datetime(0) NOT NULL,
  `ext` varchar(100) CHARACTER SET utf8mb3 COLLATE utf8mb3_general_ci NULL DEFAULT NULL,
  PRIMARY KEY (`id`) USING BTREE,
  UNIQUE INDEX `ux_undo_log`(`xid`, `branch_id`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 1 CHARACTER SET = utf8mb3 COLLATE = utf8mb3_general_ci ROW_FORMAT = Dynamic;

SET FOREIGN_KEY_CHECKS = 1;

订单和库存表:

SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;

-- ----------------------------
-- Table structure for it_order
-- ----------------------------
DROP TABLE IF EXISTS `it_order`;
CREATE TABLE `it_order`  (
  `id` bigint(0) NOT NULL AUTO_INCREMENT,
  `order_name` varchar(64) CHARACTER SET utf8mb3 COLLATE utf8mb3_general_ci NULL DEFAULT NULL,
  `order_status` char(1) CHARACTER SET utf8mb3 COLLATE utf8mb3_general_ci NULL DEFAULT NULL,
  PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB CHARACTER SET = utf8mb3 COLLATE = utf8mb3_general_ci ROW_FORMAT = Dynamic;

-- ----------------------------
-- Table structure for it_stock
-- ----------------------------
DROP TABLE IF EXISTS `it_stock`;
CREATE TABLE `it_stock`  (
  `id` bigint(0) NOT NULL AUTO_INCREMENT,
  `order_id` bigint(0) NULL DEFAULT NULL,
  `repertory` int(0) NULL DEFAULT NULL COMMENT '库存',
  `sales` int(0) NULL DEFAULT NULL COMMENT '销量',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB CHARACTER SET = utf8mb3 COLLATE = utf8mb3_general_ci ROW_FORMAT = Dynamic;

工程结构图

在这里插入图片描述

依赖项说明

父级pom.xml

	<properties>
        <maven.compiler.source>8</maven.compiler.source>
        <maven.compiler.target>8</maven.compiler.target>
        <java.version>1.8</java.version>
        <spring.cloud.alibaba.version>2.2.7.RELEASE</spring.cloud.alibaba.version>
        <spring.boot.version>2.3.12.RELEASE</spring.boot.version>
        <spring.cloud.version>Hoxton.SR12</spring.cloud.version>
    </properties>
    
    <dependencies>
        <!--SpringBoot基本场景启动-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>

        <!--SpringBoot 测试的场景启动-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
            <exclusions>
                <exclusion>
                    <groupId>org.junit.vintage</groupId>
                    <artifactId>junit-vintage-engine</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
    </dependencies>

<dependencyManagement>
        <!--Spring Cloud alibaba的版本管理, 通过dependency完成继承-->
        <dependencies>
            <dependency>
                <groupId>com.alibaba.cloud</groupId>
                <artifactId>spring-cloud-alibaba-dependencies</artifactId>
                <version>${spring.cloud.alibaba.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>

            <!--SpringBoot的版本管理-->
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-parent</artifactId>
                <version>${spring.boot.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>


            <!--Spring Cloud的版本管理-->
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>${spring.cloud.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

2 子级别pom.xml

一些常用的依赖项,例如mysql的驱动、web启动器、Druid、Mybaits框架等等。

3 孙级别pom.xml

order和stock的pom文件都是这三个

	<dependencies>
        <!--nacos-服务注册发现-->
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
        </dependency>
        <!--seata的依赖-->
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-seata</artifactId>
        </dependency>
        <!--1. openfeign依赖 -->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-openfeign</artifactId>
        </dependency>
    </dependencies>

先来配置Order模块的配置文件

stock服务的配置文件我们就不在展示了,将下面配置文件的端口号改为8222,application.name改为seata-stock即可,其他的都是相同的。

server:
  port: 8111
spring:
#===============数据库连接=============
  datasource:
    username: root
    password: 123456
    url: jdbc:mysql://localhost:3306/seata-test?characterEncoding=utf8&useSSL=false&serverTimezone=UTC
    driver-class-name: com.mysql.cj.jdbc.Driver
    type: com.alibaba.druid.pool.DruidDataSource
  application:
    name: seata-order#order的服务名称
  cloud:
  #==============nacos的地址===============
    nacos:
      discovery:
        server-addr: 127.0.0.1:8848
        username: nacos
        password: nacos
seata:
#==============seata的nacos配置=================
  registry:
    type: nacos#这里的type等同于registry.conf文件的mode = "nacos"
    nacos:
      server-addr: 127.0.0.1:8848 #seata所在的nacos服务地址
      username: nacos
      password: nacos
  config:
    type: nacos
    nacos:
      server-addr: 127.0.0.1:8848
      username: nacos
      password: nacos
      #这里要注意,这是我们在配置config.txt文件中讲述的service.vgroupMapping.后面的自定义名称
      #要跟你在上传配置文件中的组名称一致
  tx-service-group: my_tx_group
mybatis:
  mapper-locations: classpath:mapper/*.xml
  typeAliasesPackage: com.it.entity
  configuration:
    mapUnderscoreToCamelCase: true

Fegin客户端的配置(Order服务需要,Stock不需要)

1 启动类上加上@EnableFeignClients注解
2 创建fegin包,并创建一个新的接口,StockService

//value的值是要调用的服务名称,path是接口前缀路径
@FeignClient(value = "seata-stock", path = "/stock/")
public interface StockService {
	
	//这个方法是修改库存方法
    @RequestMapping("updateStockByOrderId")
    public String updateStockByOrderId(@RequestParam("orderId") Long orderId);
}

Order服务的Controller

@RestController
@RequestMapping("/order/")
public class SeataOrderController {
    @Resource
    private ItOrderService orderService;
    
    @RequestMapping("saveOrder")
    public String saveOrder() {
        return orderService.saveOrder();
    }
}

Order服务的ServiceImpl

@Service
public class ItOrderServiceImpl extends ServiceImpl<ItOrderMapper, ItOrder> implements ItOrderService {

    @Resource
    private ItOrderMapper orderMapper;

    @Resource
    private StockService stockService;

    @GlobalTransactional
    @Override
    public String saveOrder() {
        ItOrder order = new ItOrder();
        order.setOrderName("技嘉主板");
        order.setOrderStatus("2");
        orderMapper.insert(order);
        order.setId(1L);
        stockService.updateStockByOrderId(order.getId());
        int i = 1 / 0;//故意报错
        return "执行完毕!";
    }
}

Stock服务的Controller

@RestController
@RequestMapping("/stock/")
public class SeataStockController {
    @Resource
    private ItStockService stockService;

    @RequestMapping("updateStockByOrderId")
    public void updateStockByOrderId(Long orderId) {
        stockService.updateStockByOrderId(orderId);
    }
}

Stock服务的ServiceImpl

@Service
public class ItStockServiceImpl extends ServiceImpl<ItStockMapper, ItStock> implements ItStockService {

    @Resource
    private ItStockMapper stockMapper;

    @Override
    public void updateStockByOrderId(Long orderId) {
        ItStock stock = stockMapper.selectOne(new QueryWrapper<ItStock>().eq("order_id", orderId));
        stock.setRepertory(stock.getRepertory() - 1);
        stock.setSales(stock.getSales() + 1);
        stockMapper.updateById(stock);
    }
}

代码分析

我们可以看到在order服务中,我们在方法上标记了@GlobalTransactional注解,我们执行完插入语句然后远程调用修改库存服务,调用玩库存服务后故意抛出异常来测试两个服务之间是否都可以正常回滚操作

结束

最后的运行结果我就不在这里演示了,如果出现了@GlobalTransactional注解失效的问题那么有可能是以下原因:

  • Seata Server配置问题:如果Seata Server没有正确配置,包括事务组ID、事务模式等信息,那么使用@GlobalTransactional注解时就会失效。
  • 分布式事务管理器未启动:如果Seata Server未启动或者连接不上,那么使用@GlobalTransactional注解时也会失效。
  • 分布式事务传播机制问题:在分布式事务的调用链路中,每个服务都需要正确设置事务的传播机制,即在业务代码中使用@Transactional注解开启本地事务,同时在远程调用时正确传递事务上下文。如果其中任何一个服务未正确设置事务传播机制,那么@GlobalTransactional注解就会失效。
  • 事务模式不匹配:@GlobalTransactional注解的事务模式必须与Seata Server中配置的事务模式匹配,否则就会失效。如果Seata Server中配置的是AT模式,那么业务代码中使用的就必

总结

seata最难的地方是在于如何配置它,当搞清楚他的配置文件,使用它只需要一个注解即可,跟Spring提供的本地事务注解一样。
今天就分享到这里,拜拜啦!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/413868.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Docker几个概念

Docker几个概念&#xff0c;有不正确地方欢迎指正 一、首先来看一句话&#xff1a;没有Cgroups就没有LXC&#xff0c;没有LXC就没有Docker。 1、什么是Cgroup呢&#xff1f;Cgroup又名Control group&#xff0c;是Linux内核提供的一种可以限制、记录、隔离进程组所使用的物理…

VLAN 基础与划分及配置

我们都知道 VLAN 的中文名为"虚拟局域网"&#xff0c;VLAN 是一种将局域网设备从逻辑上划分成一个个不同的网段&#xff0c;从而实现虚拟工作组的新兴数据交换技术。这一新兴技术主要应用于交换机和路由器中&#xff0c;但主流应用还是在交换机之中。那今天咱们就和海…

Linux网络连接出现问题

报错截图 1.先查看NetworkManager是否启动 systemctl status NetworkManager如果输出结果中包含 "active (running)" 表示 NetworkManager 已经启动并正在运行 2.查看DNS是否配置 cat vim /etc/resolv.conf 1.查看是否有配置信息&#xff0c;如果没有请配置DNS …

华为ACL配置

模拟场景 服务器&#xff1a;192.168.3.100 销售部&#xff1a;192.168.1.1 开发部&#xff1a;192.168.2.1 模拟互联网&#xff1a;1.1.1.1 要求1&#xff1a;销售部不允许访问服务器 要求2&#xff1a;开发部可以访问服务器 要求3&#xff1a;互联网不可以访问服务器 拓扑图…

栈和队列经典题题解

目录 &#x1f349;一.括号匹配问题&#x1f349; &#x1f348;二.用队列实现栈&#x1f348; &#x1f34f;三.用栈实现队列&#x1f34f; &#x1f353;四.设计循环队列&#x1f353; &#x1f349;一.括号匹配问题&#x1f349; OJ链接力扣 题目描述&#xff1a; 思路&…

77-Linux_网络编程

网络编程一.主机字节序列和网络字节序列二.套接字地址结构1.通用socket地址结构2.专用的socket地址结构3.IP地址转换函数一.主机字节序列和网络字节序列 主机字节序列分为大端字节序和小端字节序&#xff0c;不同的主机采用的字节序列可能不同。 大端字节序是指一个整数的高位…

开店必备的5款超实用零售管理软件,第1个新手也能轻松使用!

现在越来越多的零售店老板都开始用零售管理软件来管理门店&#xff0c;提升效率&#xff0c;节约人力和时间成本。 但对于刚刚接触零售管理软件的老板来说&#xff0c;应用市场上的零售管理软件那么多&#xff0c;究竟哪些好用&#xff0c;哪些容易上手……还不太了解。 别着急…

CMake项目使用ctest+gtest进行单元测试

随着CMake工具越来越强大便捷&#xff0c;越来越多的C/C项目转而使用CMake来进行编译管理&#xff0c;它还提供了用于测试的ctest命令来执行项目中编写的单元测试。 本文就以一个实例来介绍如何使用ctest来进行单元测试。 一、环境准备 本文实例环境VSCodeMinGW64CMakegtest…

Qt程序CPU过高怎么定位解决?性能优化

自己开发的一个程序采用多线程调用url从网络上下载股票数据&#xff0c;一旦开启程序就特别的卡&#xff1b;想着优化一下&#xff1b;授之于鱼&#xff0c;不如 授之以渔&#xff1b; 1.CPU过高排查方法 &#xff08;1&#xff09;打开vs的性能探测器&#xff1b; &#xff…

Android端推送消息之极光推送

推送方式 轮询 --实现方式: 周期性主动获取网络中的数据; --缺点: 费电, 费流量; SMS --实现方式: 服务器端向手机端发送短信, 手机监听短信广播, 将拦截的短信信息进行显示; --优点: 省电, 省流量, 在没有网络的偏远地点也能接收到推送消息; --缺点: 费钱, 一毛钱一条;…

国产BI厂商:数字化时代的“卖水人”,扎根本土商业生态(上)

“没有数据&#xff0c;就没有竞争力。”随着中国经济以数字经济为重要引擎转向高质量发展道路&#xff0c;数据已成为推动经济增长的关键要素。Navesink顾问公司的创始人Thomas Redman认为&#xff0c;企业需要建立起对数据的组织和处理能力&#xff0c;只有这样才能收获大数据…

前端常用vscode插件

打开vscode配置 commandshiftP 选择Open User Setting&#xff08;首选项&#xff1a;打开用户设置(JSON)&#xff09; 1 Prettier - Code formatter "[javascript]": {"editor.defaultFormatter": "esbenp.prettier-vscode"},"[typescri…

日常记录:天梯赛练习集L1-043 阅览室

题目&#xff1a; 天梯图书阅览室请你编写一个简单的图书借阅统计程序。当读者借书时&#xff0c;管理员输入书号并按下S键&#xff0c;程序开始计时&#xff1b;当读者还书时&#xff0c;管理员输入书号并按下E键&#xff0c;程序结束计时。书号为不超过1000的正整数。当管理员…

信安大佬真的用kali吗?

Kali只是现在网络安全和kali比较火的一个操作系统 下面我为大家讲讲kali系统都有那些优点 Kali介绍Kali Linux是基于Debian的Linux发行版&#xff0c; 设计用于数字取证操作系统。面向专业的渗透测试和安全审计。 集成化&#xff1a;预装超过300个渗透测试工具兼容好&#x…

【你听说了吗】GPT-5据说已经学完了世界上现存所有的视频

文章目录前言一、GPT-5会带来什么&#xff1f;二、我们该怎么办&#xff1f;总结前言 最近半年要说最火的产品&#xff0c;无疑是ChatGPT &#xff0c;很多同学都在用 GPT 帮助自己工作&#xff0c;学习&#xff0c;提高效率&#xff01;尤其是 GPT4&#xff0c;性能强 GPT3.5…

鸟哥的Linux私房菜 学习 Shell Scripts

第十三章、学习 Shell Scripts 重点回顾 shell script 是利用 shell 的功能所写的一个『程序 (program)』&#xff0c;这个程序是使用纯文字档&#xff0c;将一些 shell 的语法与命令(含外部命令)写在里面&#xff0c; 搭配正规表示法、管线命令与数据流重导向等功能&#xf…

代码随想录算法训练营第五十三天 | 1143.最长公共子序列、1035.不相交的线、 53. 最大子序和 动态规划

打卡第53天 今日任务 1143.最长公共子序列 1035.不相交的线 53.最大子序和 动态规划 1143.最长公共子序列 给定两个字符串 text1 和 text2&#xff0c;返回这两个字符串的最长 公共子序列 的长度。如果不存在 公共子序列 &#xff0c;返回 0 。 一个字符串的 子序列 是指这…

i.MX6ULL 开发板-Mqtt 移植

转载&#xff1a;http://e.betheme.net/article/show-149636.html?actiononClick PS&#xff1a; 订阅主题&#xff0c;命令如下&#xff1a; mosquitto_sub -h localhost -t "mqtt" -v 发布主题&#xff0c;命令如下&#xff1a; mosquitto_pub -h localhost -…

96年阿里P7晒出工资单:狠补了这个,真香...

最近一哥们跟我聊天装逼&#xff0c;说他最近从阿里跳槽了&#xff0c;我问他跳出来拿了多少&#xff1f;哥们表示很得意&#xff0c;说跳槽到新公司一个月后发了工资&#xff0c;月入5万多&#xff0c;表示很满足&#xff01;这样的高薪资着实让人羡慕&#xff0c;我猜这是税后…

使用qt调用c#编写的dll库

问题背景 我需要使用qt编写界面程序来操作设备&#xff0c;设备厂家提供了一个使用C#编写的dll库&#xff0c;里面包含了各种操作设备的函数。而我不想学习C#&#xff0c;使用C来调用dll库的话&#xff0c;不论是显示调用&#xff08;提供h文件&#xff0c;dll文件&#xff09…