saga分布式事务

2023-12-10 10:06
文章标签 分布式 事务 saga

本文主要是介绍saga分布式事务,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、开篇

        在支付和交易业务中,会涉及长事务的场景。依靠单一的数据库事务无法解决整体问题,这个时候,就需要分布式事务来解决这个问题。

二、技术选型

1.saga

        Saga是一种在分布式系统中处理事务的模式,它通过将一个大的事务拆分为一系列小的、相互关联的子事务来实现。每个子事务独立执行,并且可以具有回滚和补偿机制,以保证整个事务的一致性。

子事务执行协调方式

编排模式

               

        基于事件,整个业务流程散落到各个业务系统中,比较复杂,流程难以全局理解,而且需要下游实现事件接收和发放。

控制模式

         Saga提供一个控制类,其方便参与者之前的协调工作。控制类协调整个流程,下游可以无感知迁移。

2.2pc

                                                成功情况

                                        失败情况

 

        分布式事务2PC(Two-Phase Commit)是一种用于在分布式系统中保持事务一致性的协议。它是一种基于协调者(Coordinator)和参与者(Participant)之间的交互来实现的。

        2PC协议的主要目标是在分布式环境下确保所有参与者要么都提交事务,要么都回滚事务,以保持全局事务的一致性。

三、技术实现

基于axon框架实现(控制模式)

OrderFacadeService 

订单服务facade层

/**
* 订单服务app层,1.发起创建订单 2.创建订单axon控制器。
*/
@Aggregate(cache = "orderCache")
public class OrderFacadeServiceImpl implements OrdreFacadeService{@Autowiredprivate CommandGateway commandGateWay;@Autowiredprivate PaymentFacadeService paymentClient;@Autowiredprivate LogisticsFacadeService logisticsClient;/*** 创建订单,OrderDomainService会处理对应命令*/@Overridepublic CreateOrderRespDto createOrder(CreateOrderReqDto req){var command = new CreateOrderCommand(req);return commandGateway.sendAndWait(command);}/*** 订单创建成功后创建支付*/@EventSourcingHandlerpublic void on(OrderCreatedEvent event) {var resp = paymentClient.createPay(CreatePaymentReqDto.convertTo(event));if(resp.success){// 发送支付成功事件var event= new PayOrderCreatedEvent(resp.getPaymentOrder());AggregateLifecycle.apply(event);}else{//TODO 发起创建失败事件}}/*** 支付成功后创建物流*/@EventSourcingHandlerpublic void on(PayOrderCreatedEvent event {var resp = logisticsClient.createPay(CreateLogisticsReqDto.convertTo(event));if(resp.success){// 发送创建订单结束var event= new OrderCreateEndEvent(resp.getPaymentOrder());AggregateLifecycle.apply(event);}else{//TODO 发起创建失败事件}}/***  订单创建结束后,OrderDomainService会处理对应命令*/@EventSourcingHandlerpublic void on(OrderCreateEndEvent event) {var command = new OrderCreateEndCommand(event);return commandGateway.sendAndWait(command);}}

OrderDomainService

订单领域层服务

@Service
@Aggregate(cache = "orderCache")
public class OrderDomainServiceImpl implements OrderDomainService{private OrderRepository orderRepository;/*** 创建订单*/@CommandHandler@Overridepublic OrderCreateRespDto handle(OrderCreateCommand command){Order order = Order.convertTo(command);// 存储订单orderRepository.save(order);// 发布订单已创建事件AggregateLifecycle.apply(new OrderCreatedEvent(order));return OrderCreateRespDto.convertTo(order, command);}/*** 创建订单结束*/@CommandHandler@Overridepublic CreateOrderEndRespDto handle(CreateOrderEndCommand command){Order order = Order.convertTo(command);// 更新订单orderRepository.updateCreateEnd(order);return CreateOrderEndRespDto.convertTo(order, command);}
}

todo: 在创建订单完成时候还可以再发起超时事件(延迟消息),延迟校验创建结果。

基于axon框架实现(编排模式)

OrderFacadeService 

订单服务facade层

@Aggregate(cache = "orderCache")
public class OrderFacadeServiceImpl implements OrdreFacadeService{@Autowiredprivate CommandGateway commandGateWay;/*** 创建订单,OrderDomainService会处理对应命令*/@Overridepublic CreateOrderRespDto createOrder(CreateOrderReqDto req){var command = new CreateOrderCommand(req);return commandGateway.sendAndWait(command);}/***  订单创建结束后,OrderDomainService会处理对应命令*/@EventSourcingHandlerpublic void on(OrderCreateEndEvent event) {var command = new OrderCreateEndCommand(event);return commandGateway.sendAndWait(command);}}

OrderDomainService

订单领域层服务

@Service
@Aggregate(cache = "orderCache")
public class OrderDomainServiceImpl implements OrderDomainService{private OrderRepository orderRepository;/*** 开始创建订单*/ @CommandHandler@Overridepublic OrderCreateRespDto handle(OrderCreateCommand command){Order order = Order.convertTo(command);// 存储订单orderRepository.save(order);// 发布订单已创建事件,由支付服务订阅AggregateLifecycle.apply(new OrderCreatedEvent(order));return OrderCreateRespDto.convertTo(order, command);}/** 订单创建结束*/    @CommandHandler@Overridepublic CreateOrderEndRespDto handle(CreateOrderEndCommand command){Order order = Order.convertTo(command);// 更新订单orderRepository.updateCreateEnd(order);return CreateOrderEndRespDto.convertTo(order, command);}
}

PaymentFacadeService

支付服务facade层

@Aggregate(cache = "orderCache")
public class PayFacadeServiceImpl implements PayFacadeService{@Autowiredprivate CommandGateway commandGateWay;/*** 订阅订单创建成功事件*/@EventSourcingHandlerpublic void on(OrderCreatedEvent event) {var command = new CreatePaymentCommand(event);commandGateway.sendAndWait(command);}}

PaymentDomainService

支付服务领域层

@Service
@Aggregate(cache = "orderCache")
public class PaymentDomainServiceImpl implements PaymentDomainService{private PaymentRepository paymentRepository;@CommandHandler@Overridepublic CreatePaymentEndRespDto handle(PaymentCreatCommand command){PaymentOrder order = PaymentOrder.convertTo(command);// 存储订单订单paymentRepository.save(order);// 发布已支付事件AggregateLifecycle.apply(new PaymentCreatedEvent(order));return CreatePaymentEndRespDto .convertTo(order, command);}
}

LogisticsFacadeService 和 LogisticsDomainService 

可以参考上面实现

两种模式比较

编排模式比较简单,通过一个控制器可以看整个流程,而且不需要下游接入axon框架。

这篇关于saga分布式事务的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/476792

相关文章

Nginx分布式部署流程分析

《Nginx分布式部署流程分析》文章介绍Nginx在分布式部署中的反向代理和负载均衡作用,用于分发请求、减轻服务器压力及解决session共享问题,涵盖配置方法、策略及Java项目应用,并提及分布式事... 目录分布式部署NginxJava中的代理代理分为正向代理和反向代理正向代理反向代理Nginx应用场景

Java 线程池+分布式实现代码

《Java线程池+分布式实现代码》在Java开发中,池通过预先创建并管理一定数量的资源,避免频繁创建和销毁资源带来的性能开销,从而提高系统效率,:本文主要介绍Java线程池+分布式实现代码,需要... 目录1. 线程池1.1 自定义线程池实现1.1.1 线程池核心1.1.2 代码示例1.2 总结流程2. J

Spring的基础事务注解@Transactional作用解读

《Spring的基础事务注解@Transactional作用解读》文章介绍了Spring框架中的事务管理,核心注解@Transactional用于声明事务,支持传播机制、隔离级别等配置,结合@Tran... 目录一、事务管理基础1.1 Spring事务的核心注解1.2 注解属性详解1.3 实现原理二、事务事

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

详解Spring中REQUIRED事务的回滚机制详解

《详解Spring中REQUIRED事务的回滚机制详解》在Spring的事务管理中,REQUIRED是最常用也是默认的事务传播属性,本文就来详细的介绍一下Spring中REQUIRED事务的回滚机制,... 目录1. REQUIRED 的定义2. REQUIRED 下的回滚机制2.1 异常触发回滚2.2 回

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完

Redis实现分布式锁全过程

《Redis实现分布式锁全过程》文章介绍Redis实现分布式锁的方法,包括使用SETNX和EXPIRE命令确保互斥性与防死锁,Redisson客户端提供的便捷接口,以及Redlock算法通过多节点共识... 目录Redis实现分布式锁1. 分布式锁的基本原理2. 使用 Redis 实现分布式锁2.1 获取锁

Redis分布式锁中Redission底层实现方式

《Redis分布式锁中Redission底层实现方式》Redission基于Redis原子操作和Lua脚本实现分布式锁,通过SETNX命令、看门狗续期、可重入机制及异常处理,确保锁的可靠性和一致性,是... 目录Redis分布式锁中Redission底层实现一、Redission分布式锁的基本使用二、Red

redis和redission分布式锁原理及区别说明

《redis和redission分布式锁原理及区别说明》文章对比了synchronized、乐观锁、Redis分布式锁及Redission锁的原理与区别,指出在集群环境下synchronized失效,... 目录Redis和redission分布式锁原理及区别1、有的同伴想到了synchronized关键字

分布式锁在Spring Boot应用中的实现过程

《分布式锁在SpringBoot应用中的实现过程》文章介绍在SpringBoot中通过自定义Lock注解、LockAspect切面和RedisLockUtils工具类实现分布式锁,确保多实例并发操作... 目录Lock注解LockASPect切面RedisLockUtils工具类总结在现代微服务架构中,分布