SpringCloud Alibaba微服务实战七 - 分布式事务

简介: SpringCloud Alibaba微服务实战七 - 分布式事务

导读:本篇作为SpringCloud Alibaba微服务实战系列的第七篇,主要内容是使用Seata解决分布式事务问题。系列文章,欢迎持续关注。


场景说明


订单服务order-service需要对外提供创建订单的接口,创建订单的业务逻辑如下:

先调用本地的orderService保存订单操作,然后通过feign调用远程的accout-service进行账户余额扣减,最后再通过feign调用远程的product-service进行库存扣减操作。

关键的逻辑代码如下:

  • OrderController对外提供创建订单的接口
@PostMapping("/order/create")
public ResultData<OrderDTO> create(@RequestBody OrderDTO orderDTO){
  log.info("create order:{}",orderDTO);
  orderDTO.setOrderNo(UUID.randomUUID().toString());
  orderDTO.setAmount(orderDTO.getPrice().multiply(new BigDecimal(orderDTO.getCount())));
  orderService.createOrder(orderDTO);
  return ResultData.success("create order success");
}
  • OrderServiceImpl负责处理创建订单的业务逻辑
@Transactional(rollbackFor = RuntimeException.class)
@Override
public void createOrder(OrderDTO orderDTO) {
  Order order = new Order();
  BeanUtils.copyProperties(orderDTO,order);
  //本地存储Order
  this.saveOrder(order);
  //库存扣减
  productFeign.deduct(orderDTO.getProductCode(),order.getCount());
  //账户余额扣减
  accountFeign.reduce(orderDTO.getAccountCode(), orderDTO.getAmount());
}
@Transactional(rollbackFor = RuntimeException.class)
void saveOrder(Order order) {
  orderMapper.insert(order);
}

本地先保存订单,然后调用两个远程服务进行扣减操作。

  • AccountServiceImpl扣减账户余额
@Transactional(rollbackFor = RuntimeException.class)
@Override
public void reduceAccount(String accountCode, BigDecimal amount) {
  Account account = accountMapper.selectByCode(accountCode);
  if(null == account){
    thrownew RuntimeException("can't reduce amount,account is null");
  }
  BigDecimal subAmount = account.getAmount().subtract(amount);
  if(subAmount.compareTo(BigDecimal.ZERO) < 0){
    thrownew RuntimeException("can't reduce amount,account'amount is less than reduce amount");
  }
  account.setAmount(subAmount);
  accountMapper.updateById(account);
}

做些简单的校验,当账户余额不足的时候不允许扣减操作。

  • ProductServiceImpl扣减产品库存
@Transactional(rollbackFor = RuntimeException.class)
@Override
public void deduct(String productCode, Integer deductCount) {
  Product product = productMapper.selectByCode(productCode);
  if(null == product){
    thrownew RuntimeException("can't deduct product,product is null");
  }
  int surplus = product.getCount() - deductCount;
  if(surplus < 0){
    thrownew RuntimeException("can't deduct product,product's count is less than deduct count");
  }
  product.setCount(surplus);
  productMapper.updateById(product);
}

做些简单的校验,当产品库存不足时不允许扣减操作。

order-serviceproduct-serviceaccount-service分属不同的服务,当其中一个服务抛出异常无法提交时就会导致分布式事务,如当使用feign调用account-service执行扣减账户余额时,account-service校验账户余额不足抛出异常,但是order-service的保存操作不会回滚;或者是前两步执行成功但是product-service校验不通过前面的操作也不会回滚,这就导致了数据不一致,也就是分布式事务问题!


Seata解决方案


在Springcloud Alibaba体系中使用Seata作为分布式事务解决方案,大家可以访问seata官网去了解详情。这次我们先使用Seata的file配置解决上面出现的问题,后面再来对其改造。


下载安装Seata Server。

  • 从 Release 页面下载Seata Server
  • 下载完成后直接启动Server端服务。
    在Linux/Mac下
    $ sh ./bin/seata-server.sh
    在Windows下
    bin\seata-server.bat

引入seata组件

<dependency>
  <groupId>com.alibaba.cloud</groupId>
  <artifactId>spring-cloud-alibaba-seata</artifactId>
</dependency>


在配置文件中增加seata配置

spring:
  cloud:
    alibaba:
      seata:
        tx-service-group:${spring.application.name}-seata


Seata Client 配置修改

  • 将Seata Server 配置目录下的registry.conffile.conf 2个文件拷贝到微服务中的resources文件夹下
  • 修改拷贝后的registry.conf
registry{
  type = "file"
  file {
    name = "file.conf"
  }
}
config{
  type = "file"
  file {
    name = "file.conf"
  }
}
  • 修改file.conf主要修改如下三处:
    service.vgroup_mapping.后面的值修改为配置文件spring.cloud.alibaba.seata.tx-service-group的属性
    service.default.grouplist=修改为Seata Server的ip:端口
    support.spring.datasource.autoproxy的值修改为true,开启datasource自动代理

生成undo_log表

在微服务的业务库下执行如下语句,生成undo_log表

-- 此脚本必须初始化在你当前的业务数据库中,用于AT 模式XID记录。与server端无关(注:业务数据库)
droptable`undo_log`;
CREATETABLE`undo_log` (
  `id`bigint(20) NOTNULL AUTO_INCREMENT,
  `branch_id`bigint(20) NOTNULL,
  `xid`varchar(100) NOTNULL,
  `context`varchar(128) NOTNULL,
  `rollback_info` longblob NOTNULL,
  `log_status`int(11) NOTNULL,
  `log_created` datetime NOTNULL,
  `log_modified` datetime NOTNULL,
  `ext`varchar(100) DEFAULTNULL,
  PRIMARY KEY (`id`),
  UNIQUEKEY`ux_undo_log` (`xid`,`branch_id`)
) ENGINE=InnoDB AUTO_INCREMENT=1DEFAULTCHARSET=utf8;


开启全局事务

在分布式事务方法入口添加注解@GlobalTransactional,这里只需要在createOrder方法上添加此注解即可!

@GlobalTransactional(name = "TX_ORDER_CREATE")
@Override
public void createOrder(OrderDTO orderDTO) {
  Order order = new Order();
  BeanUtils.copyProperties(orderDTO,order);
  //本地存储Order
  this.saveOrder(order);
  log.info("ORDER XID is: {}", RootContext.getXID());
  //账户余额扣减
  accountFeign.reduce(orderDTO.getAccountCode(), orderDTO.getAmount());
  //库存扣减
  productFeign.deduct(orderDTO.getProductCode(),orderDTO.getCount());
}

在代码中可以使用RootContext.getXID()获取全局xid


启动服务

服务正常启动后在Seata Server控制台可以看到注册信息


接口测试

改造完成后对接口进行测试,如果其他服务抛出异常会看到如下错误日志,再结合数据库数据观察是否正常回滚

执行过程中我们通过debug可以发现undo_log表会不断插入数据,在执行后又会被删除。

通过上面几步我们使用Seata实现了分布式事务,保证了数据的一致性,最后说一句Seata真香,你们要不要感受一下。至此本期的“SpringCloud Alibaba微服务实战七 - 分布式事务”篇也就该结束啦,咱们下期有缘再见!


目录
相关文章
|
7天前
|
存储 JavaScript 开发工具
基于HarmonyOS 5.0(NEXT)与SpringCloud架构的跨平台应用开发与服务集成研究【实战】
本次的.HarmonyOS Next ,ArkTS语言,HarmonyOS的元服务和DevEco Studio 开发工具,为开发者提供了构建现代化、轻量化、高性能应用的便捷方式。这些技术和工具将帮助开发者更好地适应未来的智能设备和服务提供方式。
30 8
基于HarmonyOS 5.0(NEXT)与SpringCloud架构的跨平台应用开发与服务集成研究【实战】
|
12天前
|
存储 SpringCloudAlibaba Java
【SpringCloud Alibaba系列】一文全面解析Zookeeper安装、常用命令、JavaAPI操作、Watch事件监听、分布式锁、集群搭建、核心理论
一文全面解析Zookeeper安装、常用命令、JavaAPI操作、Watch事件监听、分布式锁、集群搭建、核心理论。
【SpringCloud Alibaba系列】一文全面解析Zookeeper安装、常用命令、JavaAPI操作、Watch事件监听、分布式锁、集群搭建、核心理论
|
3天前
|
Java 关系型数据库 数据库
微服务SpringCloud分布式事务之Seata
SpringCloud+SpringCloudAlibaba的Seata实现分布式事务,步骤超详细,附带视频教程
15 1
|
1月前
|
存储 运维 数据可视化
如何为微服务实现分布式日志记录
如何为微服务实现分布式日志记录
63 1
|
16天前
|
存储 监控 供应链
微服务拆分的 “坑”:实战复盘与避坑指南
本文回顾了从2~3人初创团队到百人技术团队的成长历程,重点讨论了从传统JSP到前后端分离+SpringCloud微服务架构的演变。通过实际案例,总结了微服务拆分过程中常见的两个问题:服务拆分边界不清晰和拆分粒度过细,并提出了优化方案,将11个微服务优化为6个,提高了系统的可维护性和扩展性。
37 0
|
2月前
|
JSON Java 测试技术
SpringCloud2023实战之接口服务测试工具SpringBootTest
SpringBootTest同时集成了JUnit Jupiter、AssertJ、Hamcrest测试辅助库,使得更容易编写但愿测试代码。
70 3
|
2月前
|
运维 NoSQL Java
后端架构演进:微服务架构的优缺点与实战案例分析
【10月更文挑战第28天】本文探讨了微服务架构与单体架构的优缺点,并通过实战案例分析了微服务架构在实际应用中的表现。微服务架构具有高内聚、低耦合、独立部署等优势,但也面临分布式系统的复杂性和较高的运维成本。通过某电商平台的实际案例,展示了微服务架构在提升系统性能和团队协作效率方面的显著效果,同时也指出了其带来的挑战。
93 4
|
3月前
|
消息中间件 存储 负载均衡
微服务与分布式系统设计看这篇就够了!
【10月更文挑战第12天】 在现代软件架构中,微服务和分布式系统设计已经成为构建可扩展、灵活和可靠应用程序的主流方法。本文将深入探讨微服务架构的核心概念、设计原则和挑战,并提供一些关于如何在分布式系统中实现微服务的实用指导。
102 2
|
4月前
|
SpringCloudAlibaba API 开发者
新版-SpringCloud+SpringCloud Alibaba
新版-SpringCloud+SpringCloud Alibaba
|
12天前
|
SpringCloudAlibaba 负载均衡 Dubbo
【SpringCloud Alibaba系列】Dubbo高级特性篇
本章我们介绍Dubbo的常用高级特性,包括序列化、地址缓存、超时与重试机制、多版本、负载均衡。集群容错、服务降级等。
【SpringCloud Alibaba系列】Dubbo高级特性篇