从0到1 手把手搭建spring cloud alibaba 微服务大型应用框架(五) SEATA分布式事务篇(中)shardingshere 多库读写分离/分库分表下分布式事务完整代码及案例

本文涉及的产品
云原生网关 MSE Higress,422元/月
服务治理 MSE Sentinel/OpenSergo,Agent数量 不受限
任务调度 XXL-JOB 版免费试用,400 元额度,开发版规格
简介: 从0到1 手把手搭建spring cloud alibaba 微服务大型应用框架(五) SEATA分布式事务篇(中)shardingshere 多库读写分离/分库分表下分布式事务完整代码及案例

image.png 本篇紧接上文《从0到1 手把手搭建spring cloud alibaba 微服务大型应用框架(五) SEATA分布式事务篇(上) 运行原理以及AT模式源码启动版集成》


读写分离/分库分表 配置集成


参考我之前的该篇文章《从0到1 手把手搭建spring cloud alibaba 微服务大型应用框架(四) (mini-cloud) 集成shardingSphere 读写分离 》


mini-cloud下构建分布式事务案例场景-电商订单-库存分布式事务问题


电商订单-库存分布式事务场景图示


1.png

场景描述

订单库存是典型的微服务分布式事务问题,首先看看如上图所示的流程


1.用户下单,订单服务首先调用库存服务校验以及扣除库存


2.如果扣除成功,订单服务则创建一个订单并返回用户提示下单成功


3.如果扣除失败,订单服务则返回用户提示库存不足,下单失败。


场景很简单,正常情况下只要业务数据逻辑正确,本身是没有什么大问题的,


但是注意看红框框住的部分,如果扣除库存成功了,但是订单服务后续业务处理报错了,那么将会是什么结果呢,首先订单服务本身的数据肯定是会回滚的,因为在一个事务里报错了,但是库存服务的数据已经commit了,所以现在就会造成库存多扣了实际订单没有创建的情况,其实实际情况要复杂得多,因为图示是简化版,真实项目里可能会调用了更多的服务,相关联的数据有可能横跨很多数据库,所以分布式事务的控制是很必要的,其实现在有很多分布式事务的解决方案

1. 最简单的是xa分布式事务方案,因为是基于数据库本身的机制进行的,少许代码即可


2.二段式,三段式,这些就和业务挂钩相对紧密了,需要手动去写业务回滚的业务处理部分


3.saga 本身其实也是需要手动写业务归滚部分代码,但是现如今框架会帮助处理大部分逻辑流程


4.最终一致性,一般应用mq消息中间件,业务发起到结束期间允许数据的不一致,最后结束时保证一直,这种情况一般不会发生在强一直行场景下,比如转账,支付,扣除库存等,所以这里不太适合,但是可以配和缓存做伪强一致性处理,后续可以单独写一篇高并发下订单问题篇


4.at 模式,比如seata的at模式,框架本身会帮助你拦截你本次涉及的所有服务的数据库sql,并生成反向sql保存起来,如果成功则删除这些sql,失败就执行这些sql恢复数据,可以参考上一篇,里面对原理有所介绍,下面图是简单画的seata at模式原理图


从0到1 手把手搭建spring cloud alibaba 微服务大型应用框架(五) SEATA分布式事务篇(上) 运行原理以及AT模式源码启动版集成

image.png

mini-cloud 集成订单服务与库存服务(shardingsphere读写分离)


创建订单服务和库存服务


在mini-cloud 框架中有一个simulate模块,专门用来创建各个业务创建的模拟服务,这里我们直接创建两个服务,订单服务和库存服务,过程不细说了,就是正常创建服务并且注册到我们的nacos里面,可以参考我之前的文章,结果如下

1.png

创建订单服务和库存服务数据库

由于我们是三个库的读写分离,简化起见只展示库和表设计

商品库存库 与表设计

1.png

1.png

订单库与表设计

1.png

1.png

为了简单起见每个库只有一个表,每个表就俩字段,undo_log是后面seata 需要的回滚表,后续回有介绍


创建订单服务和库存服务配置中心内sharesphere 读写分离


由于我们是shareshpere的读写分离库,所以需要配置shareshpere 读写分离并写入nacos配置中心


商品库存库


spring:
  shardingsphere:
    props:
      sql:
        show: true
    datasource:
      names: master,slave0,slave1
      master:
        type: com.alibaba.druid.pool.DruidDataSource
        driver-class-name: com.mysql.cj.jdbc.Driver
        url: jdbc:mysql://${MYSQL_HOST:master}:${MYSQL_PORT:3306}/${MYSQL_DB:mini_cloud_goods}?characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=false&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=GMT%2B8&allowMultiQueries=true&allowPublicKeyRetrieval=true
        username: root
        password: root
      slave0:
        type: com.alibaba.druid.pool.DruidDataSource
        driver-class-name: com.mysql.cj.jdbc.Driver
        url: jdbc:mysql://${MYSQL_HOST:slave0}:${MYSQL_PORT:3306}/${MYSQL_DB:mini_cloud_goods}?characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=false&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=GMT%2B8&allowMultiQueries=true&allowPublicKeyRetrieval=true
        username: root
        password: root
      slave1:
        type: com.alibaba.druid.pool.DruidDataSource
        driver-class-name: com.mysql.cj.jdbc.Driver
        url: jdbc:mysql://${MYSQL_HOST:slave1}:${MYSQL_PORT:3306}/${MYSQL_DB:mini_cloud_goods}?characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=false&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=GMT%2B8&allowMultiQueries=true&allowPublicKeyRetrieval=true
        username: root
        password: root
    sharding:
      master-slave-rules:
        master:
          master-data-source-name: master
          slave-data-source-names: slave0,slave1
  application:
    name: @artifactId@
  cloud:
    nacos:
      discovery:
        server-addr: ${NACOS_HOST:127.0.0.1}:${NACOS_PORT:8848}
      config:
        server-addr: ${spring.cloud.nacos.discovery.server-addr}
        file-extension: yml
        shared-configs:
          - application-${spring.profiles.active}.${spring.cloud.nacos.config.file-extension}
  profiles:
    active: @profiles.active@

订单库

spring:
  shardingsphere:
    props:
      sql:
        show: true
    datasource:
      names: master,slave0,slave1
      master:
        type: com.alibaba.druid.pool.DruidDataSource
        driver-class-name: com.mysql.cj.jdbc.Driver
        url: jdbc:mysql://${MYSQL_HOST:master}:${MYSQL_PORT:3306}/${MYSQL_DB:mini_cloud_order}?characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=false&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=GMT%2B8&allowMultiQueries=true&allowPublicKeyRetrieval=true
        username: root
        password: root
      slave0:
        type: com.alibaba.druid.pool.DruidDataSource
        driver-class-name: com.mysql.cj.jdbc.Driver
        url: jdbc:mysql://${MYSQL_HOST:slave0}:${MYSQL_PORT:3306}/${MYSQL_DB:mini_cloud_order}?characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=false&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=GMT%2B8&allowMultiQueries=true&allowPublicKeyRetrieval=true
        username: root
        password: root
      slave1:
        type: com.alibaba.druid.pool.DruidDataSource
        driver-class-name: com.mysql.cj.jdbc.Driver
        url: jdbc:mysql://${MYSQL_HOST:slave1}:${MYSQL_PORT:3306}/${MYSQL_DB:mini_cloud_order}?characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=false&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=GMT%2B8&allowMultiQueries=true&allowPublicKeyRetrieval=true
        username: root
        password: root
    sharding:
      master-slave-rules:
        master:
          master-data-source-name: master
          slave-data-source-names: slave0,slave1
  application:
    name: @artifactId@
  cloud:
    nacos:
      discovery:
        server-addr: ${NACOS_HOST:127.0.0.1}:${NACOS_PORT:8848}
      config:
        server-addr: ${spring.cloud.nacos.discovery.server-addr}
        file-extension: yml
        shared-configs:
          - application-${spring.profiles.active}.${spring.cloud.nacos.config.file-extension}
  profiles:
    active: @profiles.active@
security:
  enable: false
  client:
    ignore-urls:
      - /order/**

编写订单服务和库存服务相互调用代码


库存服务


fegin 部分

1.png

package com.minicloud.goods.fegin;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
@FeignClient(contextId = "remoteSimulateGoodsService",value = "mini-cloud-simulate-goods-biz")
public interface RemoteSimulateGoodsService {
    @PostMapping("/goods/subStock/{goodsId}/{num}")
    boolean subStock(@PathVariable("goodsId")Integer goodsId, @PathVariable("num")Integer num);
}

业务部分


GoodsController

package com.minicloud.goods.controller;
import com.minicloud.goods.service.GoodsService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("goods")
public class GoodsController {
    @Autowired
    private GoodsService goodsService;
    /**
     * @desc: 模拟扣减商品库存
     * @param goodsId 商品id
     * @param num 商品数量
     *
     * */
    @PostMapping("/subStock/{goodsId}/{num}")
    public boolean subStock(@PathVariable("goodsId")Integer goodsId,@PathVariable("num")Integer num){
        boolean result = goodsService.subStock(goodsId,num);
        if(!result){
            return false;
        }else {
            return true;
        }
    }
}

GoodsServiceImpl

package com.minicloud.goods.service.impl;
import cn.org.atool.fluent.mybatis.base.crud.IUpdate;
import com.minicloud.goods.entity.GoodsEntity;
import com.minicloud.goods.mapper.GoodsMapper;
import com.minicloud.goods.service.GoodsService;
import com.minicloud.goods.wrapper.GoodsQuery;
import com.minicloud.goods.wrapper.GoodsUpdate;
import io.seata.core.context.RootContext;
import io.seata.spring.annotation.GlobalTransactional;
import lombok.extern.slf4j.Slf4j;
import org.apache.shardingsphere.transaction.annotation.ShardingTransactionType;
import org.apache.shardingsphere.transaction.core.TransactionType;
import org.apache.shardingsphere.transaction.core.TransactionTypeHolder;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
/**
 * @Author alan.wang
 * @date: 2022-02-17 16:11
 */
@Slf4j
@Service
public class GoodsServiceImpl implements GoodsService {
    @Autowired
    private GoodsMapper goodsMapper;
    @Override
    @Transactional(rollbackFor = Exception.class)
    public boolean subStock(Integer goodsId, Integer num) {
        log.info("开始全局事务,XID = " + RootContext.getXID());
        IUpdate iUpdate = new GoodsUpdate().set.goodsStock().applyFunc("goods_stock -"+num).end().where.goodsId().eq(goodsId).end();
        goodsMapper.updateBy(iUpdate);
        GoodsEntity goodsEntity = goodsMapper.findById(goodsId);
        if(goodsEntity.getGoodsStock()>0){
            log.info("库存扣除成功 ,goodsId:{},num:{}",goodsId,num);
            return true;
        }else {
            log.info("库存扣除失败 ,goodsId:{},num:{}",goodsId,num);
            return false;
        }
    }
}

订单服务


OrderController

package com.minicloud.simulate.order.controller;
import com.minicloud.simulate.order.service.OrderService;
import io.seata.spring.annotation.GlobalTransactional;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.PutMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/order")
public class OrderController {
    @Autowired
    private OrderService orderService;
    /**
     * @desc: 模拟生成订单
     * **/
    @PutMapping("/create")
    public ResponseEntity createOrder() throws Exception {
        orderService.newOrder();
        return ResponseEntity.ok().build();
    }
}

OrderServiceImpl

package com.minicloud.simulate.order.service.impl;
import com.minicloud.goods.fegin.RemoteSimulateGoodsService;
import com.minicloud.simulate.order.entity.OrdersEntity;
import com.minicloud.simulate.order.mapper.OrdersMapper;
import com.minicloud.simulate.order.service.OrderService;
import io.seata.core.context.RootContext;
import io.seata.spring.annotation.GlobalTransactional;
import lombok.extern.slf4j.Slf4j;
import org.apache.shardingsphere.transaction.annotation.ShardingTransactionType;
import org.apache.shardingsphere.transaction.core.TransactionType;
import org.apache.shardingsphere.transaction.core.TransactionTypeHolder;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
import javax.annotation.Resource;
import java.time.LocalDateTime;
@Slf4j
@Service
public class OrderServiceImpl implements OrderService {
    @Autowired
    private OrdersMapper ordersMapper;
    @Resource
    private RemoteSimulateGoodsService remoteSimulateGoodsService;
    @Override
    @Transactional(rollbackFor = Exception.class)
    @ShardingTransactionType(TransactionType.BASE)
    public String newOrder() throws Exception {
        log.info("开始全局事务,XID = " + RootContext.getXID());
         //demo 测试使用,实际应该用有意义的分布式id
        String orderId = System.nanoTime()+"";
        OrdersEntity ordersEntity = new OrdersEntity();
        ordersEntity.setOrderId(orderId);
        ordersEntity.setOrderCreateTime(LocalDateTime.now());
        ordersMapper.insertWithPk(ordersEntity);
        boolean result = remoteSimulateGoodsService.subStock(1,3);
        if(result){
            log.info("创建订单成功 ,id:{}",orderId);
        }else {
            throw new Exception("库存不足,创建订单失败");
        }
        return orderId;
    }
}

启动服务并测试正常情况


首先启动所有服务,注册中心,认证中心,用户中心,网关,订单服务,库存服务,并查看

1.png

nacos服务注册情况

image.png

我们首先在库存数据库创建一个商品并且赋值初始库存为10

1.png

1.png

然后用请求创建订单接口,针对goods_id是1的商品扣除3个,正常情况下应该是库存少3变为7

然后订单表里多一条数据,那么我们测一下

1.png

运行结果:

1.png

库存表确实变为了7

1.png

订单表创建了一条数据

1.png

篇幅原因,下篇介绍异常情况下数据不一致以及seata at 的强一致性回滚

相关实践学习
每个IT人都想学的“Web应用上云经典架构”实战
本实验从Web应用上云这个最基本的、最普遍的需求出发,帮助IT从业者们通过“阿里云Web应用上云解决方案”,了解一个企业级Web应用上云的常见架构,了解如何构建一个高可用、可扩展的企业级应用架构。
MySQL数据库入门学习
本课程通过最流行的开源数据库MySQL带你了解数据库的世界。   相关的阿里云产品:云数据库RDS MySQL 版 阿里云关系型数据库RDS(Relational Database Service)是一种稳定可靠、可弹性伸缩的在线数据库服务,提供容灾、备份、恢复、迁移等方面的全套解决方案,彻底解决数据库运维的烦恼。 了解产品详情: https://www.aliyun.com/product/rds/mysql 
相关文章
|
2月前
|
SQL Java 数据库连接
Spring Data JPA 技术深度解析与应用指南
本文档全面介绍 Spring Data JPA 的核心概念、技术原理和实际应用。作为 Spring 生态系统中数据访问层的关键组件,Spring Data JPA 极大简化了 Java 持久层开发。本文将深入探讨其架构设计、核心接口、查询派生机制、事务管理以及与 Spring 框架的集成方式,并通过实际示例展示如何高效地使用这一技术。本文档约1500字,适合有一定 Spring 和 JPA 基础的开发者阅读。
282 0
|
3月前
|
监控 Java API
Spring Boot 3.2 结合 Spring Cloud 微服务架构实操指南 现代分布式应用系统构建实战教程
Spring Boot 3.2 + Spring Cloud 2023.0 微服务架构实践摘要 本文基于Spring Boot 3.2.5和Spring Cloud 2023.0.1最新稳定版本,演示现代微服务架构的构建过程。主要内容包括: 技术栈选择:采用Spring Cloud Netflix Eureka 4.1.0作为服务注册中心,Resilience4j 2.1.0替代Hystrix实现熔断机制,配合OpenFeign和Gateway等组件。 核心实操步骤: 搭建Eureka注册中心服务 构建商品
615 3
|
1月前
|
消息中间件 缓存 Java
Spring框架优化:提高Java应用的性能与适应性
以上方法均旨在综合考虑Java Spring 应该程序设计原则, 数据库交互, 编码实践和系统架构布局等多角度因素, 旨在达到高效稳定运转目标同时也易于未来扩展.
111 8
|
3月前
|
Java 应用服务中间件 开发者
Spring Boot 技术详解与应用实践
本文档旨在全面介绍 Spring Boot 这一广泛应用于现代企业级应用开发的框架。内容将涵盖 Spring Boot 的核心概念、核心特性、项目自动生成与结构解析、基础功能实现(如 RESTful API、数据访问)、配置管理以及最终的构建与部署。通过本文档,读者将能够理解 Spring Boot 如何简化 Spring 应用的初始搭建和开发过程,并掌握其基本使用方法。
322 2
|
3月前
|
人工智能 监控 安全
如何快速上手【Spring AOP】?核心应用实战(上篇)
哈喽大家好吖~欢迎来到Spring AOP系列教程的上篇 - 应用篇。在本篇,我们将专注于Spring AOP的实际应用,通过具体的代码示例和场景分析,帮助大家掌握AOP的使用方法和技巧。而在后续的下篇中,我们将深入探讨Spring AOP的实现原理和底层机制。 AOP(Aspect-Oriented Programming,面向切面编程)是Spring框架中的核心特性之一,它能够帮助我们解决横切关注点(如日志记录、性能统计、安全控制、事务管理等)的问题,提高代码的模块化程度和复用性。
|
3月前
|
安全 算法 Java
在Spring Boot中应用Jasypt以加密配置信息。
通过以上步骤,可以在Spring Boot应用中有效地利用Jasypt对配置信息进行加密,这样即使配置文件被泄露,其中的敏感信息也不会直接暴露给攻击者。这是一种在不牺牲操作复杂度的情况下提升应用安全性的简便方法。
916 10
|
4月前
|
NoSQL Java Redis
Redis基本数据类型及Spring Data Redis应用
Redis 是开源高性能键值对数据库,支持 String、Hash、List、Set、Sorted Set 等数据结构,适用于缓存、消息队列、排行榜等场景。具备高性能、原子操作及丰富功能,是分布式系统核心组件。
525 2
|
4月前
|
安全 Java Nacos
0代码改动实现Spring应用数据库帐密自动轮转
Nacos作为国内被广泛使用的配置中心,已经成为应用侧的基础设施产品,近年来安全问题被更多关注,这是中国国内软件行业逐渐迈向成熟的标志,也是必经之路,Nacos提供配置加密存储-运行时轮转的核心安全能力,将在应用安全领域承担更多职责。
|
4月前
|
Java Linux 网络安全
Linux云端服务器上部署Spring Boot应用的教程。
此流程涉及Linux命令行操作、系统服务管理及网络安全知识,需要管理员权限以进行配置和服务管理。务必在一个测试环境中验证所有步骤,确保一切配置正确无误后,再将应用部署到生产环境中。也可以使用如Ansible、Chef等配置管理工具来自动化部署过程,提升效率和可靠性。
484 13
|
5月前
|
Java 数据库 开发者
Spring Boot 框架超级详细总结及长尾关键词应用解析
本文深入讲解Spring Boot框架的核心概念、功能特性及实际应用,涵盖自动配置、独立运行、starter依赖等优势。通过Web开发、微服务架构、批处理等适用场景分析,结合在线书店实战案例,演示项目初始化、数据库设计、分层架构实现全流程。同时探讨热部署、多环境配置、缓存机制与事务管理等高级特性,助你高效掌握Spring Boot开发技巧。代码示例详尽,适合从入门到进阶的学习者。
2062 0

热门文章

最新文章