【Seata】分布式事务解决方案——理论

目录

  • 回顾
    • 什么是事务
    • 数据库事务的四大特性:ACID
  • 分布式事务
    • 解释
    • 面临挑战
    • 分布式事务产生场景
      • 1. 单体架构中多数据源场景
      • 2. 分布式架构场景
    • 分布式事务解决方案
      • jta + Atomikos
      • LCN模式
        • 问题
      • Alibaba的Seata解决分布式事务
        • 问题
      • 使用MQ解决分布式事务问题
        • 问题1:关于消息丢失问题(消息可靠性)
        • 问题2:事务回滚怎么处理

回顾

什么是事务

  • 事务可以看做是一次大的活动,它由不同的小活动组成,这些活动要么全部成功,要么全部失败。

数据库事务的四大特性:ACID

  • A(Atomic):原子性,构成事务的所有操作,要么都执行完成,要么全部不执行,不可能出现部分成功部分失败的情况。

  • C(Consistency):一致性,在事务执行前后,数据库的一致性约束没有被破坏。比如:张三向李四转 100 元,转账前和转账后的数据是正确状态这叫一致性,如果出现张三转出 100 元,李四账户没有增加 100 元这就出现了数 据错误,就没有达到一致性。

  • I(Isolation):隔离性,数据库中的事务一般都是并发的,隔离性是指并发的两个事务的执行互不干扰,一个事务不能看到其他事务的运行过程的中间状态。通过配置事务隔离级别可以比避免脏读、重复读问题。

  • D(Durability):持久性,事务完成之后,该事务对数据的更改会持久到数据库,且不会被回滚。

数据库事务在实现时会将一次事务的所有操作全部纳入到一个不可分割的执行单元,该执行单元的所有操作要么都成功,要么都失败,只要其中任一操作执行失败,都将导致整个事务的回滚。

分布式事务

解释

  • 分布式事务是指涉及多个独立数据库或资源的事务处理。
  • 在分布式系统中,每个数据库或资源都是独立的,它们可能位于不同的物理位置,由不同的机器或进程管理。
  • 分布式事务的目标是确保在所有参与的数据库或资源上,事务要么全部提交成功,要么全部回滚失败,以保持整个系统的一致性。

面临挑战

  • 分布式事务面临的挑战是如何在分散的环境中协调各个参与者的操作,并确保事务的原子性、一致性、隔离性和持久性。
  • 为了解决这个问题,通常使用两阶段提交(Two-Phase Commit,2PC)协议或者三阶段提交(Three-Phase Commit,3PC)协议等机制来实现分布式事务的管理。

在分布式系统中,分布式事务的设计和实现需要考虑到网络延迟、节点故障、并发访问等问题。因此,合理的分布式事务设计是提高系统稳定性和性能的关键。

分布式事务产生场景

1. 单体架构中多数据源场景

假设有一个电子商务应用,其中包含订单服务和支付服务,订单数据存储在MySQL数据库中,而支付数据存储在另一个MongoDB数据库中。在单体架构中,订单服务和支付服务分别是独立的模块,各自维护自己的数据源。

在这个场景下,多数据源场景可能导致分布式事务问题的流程如下:

  1. 用户在下订单时,订单服务会创建一个订单,并向MySQL数据库插入订单数据。

  2. 同时,订单服务需要调用支付服务来完成支付操作。支付服务会从MongoDB数据库中查询用户的支付信息,根据订单金额进行支付,并将支付结果存储到MongoDB数据库中。

  3. 在事务开始之前,订单服务和支付服务都需要获取一个数据库连接,开启一个数据库事务。

  4. 订单服务在MySQL数据库中插入订单数据,并等待支付服务完成支付。

  5. 支付服务从MongoDB数据库中获取支付信息,并进行支付操作。

  6. 如果支付操作成功,支付服务将支付结果保存到MongoDB数据库中。

  7. 如果支付操作失败,支付服务将回滚支付操作,并且回滚MongoDB数据库中的变更。

  8. 然后,支付服务将事务提交,关闭数据库连接。

  9. 订单服务等待支付服务的响应。

  10. 如果支付服务成功完成支付并提交事务,订单服务也提交事务,订单和支付数据都保存成功。

  11. 如果支付服务执行失败或超时,订单服务将回滚订单操作,并且回滚MySQL数据库中的变更。

  12. 订单服务提交失败,订单和支付数据都没有保存成功。

在这个流程中,由于订单服务和支付服务分别维护自己的数据源,它们在进行跨数据源的操作时面临分布式事务问题。如果支付服务在支付操作后发生故障,导致支付服务无法提交事务,则订单服务的订单数据已经插入到MySQL数据库中,但支付数据未能成功保存到MongoDB数据库中,这就导致了数据的不一致。

2. 分布式架构场景

  • 简单来说:跨JVM调用接口,一定会发生分布式事务问题。
  • 在RPC接口调用的过程中,A调用B服务接口之后,当A接口报错,无法回滚B接口的事务,导致最终A接口事务回滚了,B接口事务没有回滚,因此就需要解决分布式事务问题。

分布式事务解决方案

  1. 单体架构中多数据源场景可以使用jta + Atomikos。
  2. 基于RabbitMQ的形式解决,最终一致性的思想。
  3. 基于RocketMQ解决,采用事务消息。
  4. LCN:采用LCN模式,假关闭连接(目前已经被淘汰)。
  5. Alibaba的Seata背景非常强大,已经成为了主流,但是性能一般。如果你想要解决分布式事务问题,又想接口快速响应,就不要用Seata,Seata会导致接口响应变慢,就会发生超时。如果项目是追求快速响应,建议使用MQ最终一致性方案。

jta + Atomikos

JTA(Java Transaction API)和Atomikos是两个常用的工具,可以用于解决分布式事务问题。

  1. 首先,你需要在应用程序中引入JTA和Atomikos的依赖。你可以通过Maven或Gradle等构建工具来添加这些依赖。

  2. 然后,你需要在应用程序中配置Atomikos事务管理器。配置包括数据库连接池的设置以及Atomikos事务管理器的一些属性,如事务超时时间等。

  3. 在应用程序中,你可以使用JTA注解或编程式方式来标记事务。使用JTA注解可以在方法或类级别上标记事务,而编程式方式需要在代码中显式地开启、提交或回滚事务。

  4. 如果你的应用程序需要与其他应用程序或服务进行交互,你可以使用Atomikos的分布式事务协调器来确保所有的事务操作在一个统一的事务中执行。分布式事务协调器可以通过一些配置文件中的信息来管理分布式事务。

  5. 当一个分布式事务执行完成后,你可以使用Atomikos提供的API来查看事务的状态以及处理异常。你可以根据需要进行回滚、提交或恢复事务。

LCN模式

在这里插入图片描述

  1. 发起方和参与方与我们的LCN管理器全局事务协调者一直保持长连接。
  2. 发起方在调用接口之前会使用AOP生成一个全局的事务分组ID。
  3. 发起方在调用接口的时候会在请求头中传递该全局事务分组ID。
  4. 参与方会从请求头中获取该事务分组ID,当前业务执行完毕之后不会提交该事务,则会使用假关闭。
  5. 发起方调用接口完成之后,如果出现异常的情况下,会通知事务协调者回滚该事务,协调者再通知参与方回滚事务,这样两个服务都发生了事务回滚。
问题

LCN基于数据源假关闭,事务如果不提交,有可能会导致Mysql innoDB存储引擎的行锁机制,因为你事务没提交,行锁就不会释放。而且积分服务不提交事务,其它项目查询积分根本查不到你未提交的数据。

Alibaba的Seata解决分布式事务

Seata为用户提供了AT、TCC、SAGA和XA事务模式,为用户打造一站式分布式事务解决方案。其实说起来Seata和LCN差不多
在这里插入图片描述

  • 当发起方调用参与方接口,支付服务首先更新状态为已支付,那么这时undolog日志会生成前置镜像和后置镜像,前置镜像就是修改前的状态。这时调用积分服务,积分服务insert一条积分数据,undolog日志生成前置镜像和后置镜像。

  • 那么调用已完成,这时支付服务方报错了,那么通过undolog日志会进行逆向回滚,回滚到之前的状态,然后通知事务协调者,告诉积分服务也要回滚,那么积分服务的undolog日志也进行了逆向回滚,insert的逆向就是delete,这样两个服务就都回滚成功了。

  • 那么Seata与LCN最大的区别就是,在支付服务调用积分服务接口时,积分服务insert一条数据,LCN是不提交事务的,这样就导致别人无法查询到这条新insert的数据。但Seata提交了事务,所以可以查到。

问题
  • 有可能存在短暂脏读问题。
    • 解释一下脏读:一个事务没有提交的数据被另外一个事务读到,而全局事务是多个本地事务的集合,如果在全局事务下,某个本地事务提交了,如果没有全局控制,那么这个提交的事务也有可能被其他事务读到,也是一种脏写。
  • 这个问题的本质是:全局事务中的某个本地事务完成不代表全局事务也完成。
  • 比如我们支付服务调用积分服务,积分服务insert后提交事务,再相应给支付服务,这时支付服务报错了,打算回滚自己和积分服务,但这时其它的服务是可以查到积分服务刚insert的这条数据。但这时又回滚了,再查询发现刚查到的这条数据又没了

使用MQ解决分布式事务问题

在这里插入图片描述

  1. 当支付服务更新状态为已完成,然后发送消息到MQ,
  2. 积分服务获取到消息再insert一条数据
问题1:关于消息丢失问题(消息可靠性)
  1. ACK确认机制:生产者必须确保消息投递到MQ成功
    • 如果生产者投递消息失败的情况下,则通过日志记录下来,后期通过定时任务自动补偿。
  2. 服务端消息持久化,避免MQ宕机之后消息丢失
    • 生产者发送消息到MQ服务端,服务端持久化到硬盘上,再回执给生产者,告诉它消息投递成功。
  3. 重试机制:确保消费者消息消费成功(同时注意幂等性问题)
    • 如果消费者消费失败的情况下则MQ会采用间隔的形式不断重试,重试过程中需要解决幂等性问题。
问题2:事务回滚怎么处理
  • MQ自身无法回滚,只有补偿。
  • 如果支付服务方出错,这时积分服务已经insert并提交事务,那是系统的问题,积分增加就增加了,或者之后再扣除。
  • 如果你积分服务出错了,那后续补偿再把积分补上

一个原则:最终一致性

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

连接智能世界,Vatee万腾平台在行动

随着科技的飞速发展,我们生活在一个日益智能化的世界里。从智能家居到智慧城市,从无人驾驶到远程医疗,智能科技正以前所未有的速度改变着我们的生活。而在这场智能革命中,Vatee万腾平台凭借其前瞻性的视野和创新的实力&#xff0c…

企业级WordPress开发 – 创建企业级网站的优秀技巧

目录 1 “企业级”一词是什么意思? 2 使用 WordPress 进行企业级 Web 开发有哪些好处? 3 使用 WordPress 进行企业级开发的主要好处 3.1 WordPress 可扩展、灵活且价格实惠 3.2 WordPress 提供响应式 Web 开发 3.3 WordPress 提供巨大的可扩展…

全球进口家装水管十大品牌

随着科技进步发展,健康环保家居观念深入人心,家装管道用水安全对人们健康起来至关重要的作用,水管管道业主非常重视,但面临市场参差不齐的进口水管不知如何选择,下面就和大家推荐一下进口家装水管十大品牌 1.德国洁水…

未授权访问漏洞总结

以下总结了常见的未授权访问漏洞,还在持续更新中,遇到就会补充。欢迎大家关注~ 目录 FTP未授权访问(21) 漏洞原理 漏洞检测 漏洞利用 漏洞修复 LDAP未授权访问(389) 漏洞原理 漏洞检测 漏洞利用 …

MacOS - 启动台多了个『卸载 Adobe Photoshop』

问题描述 今天安装好了 Adobe Ps,但是发现启动台多了个『卸载 Adobe Photoshop』强迫症又犯了,想把它干掉! 解决方案 打开访达 - 前往 - 资源库,搜索要卸载的名字就可以看到,然后移除到垃圾筐

Javaweb07-JavaBean技术和Jsp开发模式

JavaBean技术和Jsp开发模式 一.JavaBean技术 1.JavaBean的基本概念 **JavaBean组件:**与html分离且使用Java代码封装类 **JavaBean分类:**可视化JavaBean:swing 非可视化JavaBean:用于封装实体和业务逻辑 JavaBean特点&#x…

if/case条件测试语句

一 条件测试 1.1返回码 $? $? 返回码 用来哦按段命令或者脚本是否执行成功 0 true为真就是成功成立 非0 false 失败或者异常 1.2 test 命令 可以进行条件测试 然后根据返回值来判断条件是否成立 -e :exist 测试目录或者目录是否存在 -d : director…

华为北向网管NCE开发教程8(性能数据)

1接口说明 通过北向网管获取性能数据的接口,主要用PerformanceManagementMgr_I。 获取当前:getAllCurrentPMData 获取历史:getHistoryPMData Common_IHolder commonHolder new Common_IHolder(); emsSession.getManager("Performance…

itsm服务管理工具有哪些?

itsm(IT服务管理)是现代企业管理IT服务的关键框架,可帮助组织提供高效的IT服务,并确保业务持续运行。使用itsm服务管理工具是实现这一目标的关键,下面我们来看看itsm服务管理工具有哪些: 1. ServiceDesk Pl…

MyBatis中获取Mysql数据库插入记录的主键值

在MyBatis中,你可以使用多种方式获取插入记录的主键值。以下是常见的几种方法: 1. 使用 useGeneratedKeys 和 keyProperty 这是MyBatis提供的最直接的方法。在你的Mapper XML文件中,使用 useGeneratedKeys 和 keyProperty 来指定MyBatis在插…

为啥找对象千万别找大厂男,还好我不是大厂的。。

网上看到一大厂女员工发文说:找对象千万别找大厂男,理由说了一大堆,无非就是大厂男为了逃避带娃,以加班为由宁愿在工位上玩游戏也不愿回家。当然这种观点有的人赞同有的人反对。 网友精彩评论: --------------下面是今…

通过visualVM远程监控java进程

1. VisualVM 简介及功能介绍 VisualVM 是一个功能强大的工具,用于监视、分析、配置和调试 Java 应用程序的运行状态。它提供了丰富的功能来帮助开发者和运维人员理解和优化 Java 应用程序的性能。VisualVM 集成了多种 JDK 工具,并通过直观的 GUI 界面提供…

Hi3861 OpenHarmony嵌入式应用入门--ADC

本篇讲解使用ADC进行采样,并使用API将采样值转为电压。 电路原理图 通过hi-12f_v1.1.2-规格书-20211202.pdf 找到IO9对应的ADC通道 GPIO API API名称 说明 hi_u32 hi_gpio_init(hi_void); GPIO模块初始化 hi_u32 hi_gpio_set_dir(hi_gpio_idx id, hi_gpio_dir …

论文阅读笔记:Towards Higher Ranks via Adversarial Weight Pruning

论文阅读笔记:Towards Higher Ranks via Adversarial Weight Pruning 1 背景2 创新点3 方法4 模块4.1 问题表述4.2 分析高稀疏度下的权重剪枝4.3 通过SVD进行低秩逼近4.4 保持秩的对抗优化4.5 渐进式剪枝框架 5 效果5.1 和SOTA方法对比5.2 消融实验5.3 开销分析 6 结…

如何打开mobi文件?两个步骤解决

打开MOBI格式的电子书,其实相当简便。NeatReader作为一个兼容多格式多系统的电子书阅读器,对MOBI格式的支持自然不在话下。下面是使用NeatReader阅读MOBI文件的步骤: 第一步:下载并安装NeatReader: 首先,你…

彩虹PLM:引领产品生命周期管理的创新潮流

彩虹PLM:引领产品生命周期管理的创新潮流 在当今快速发展的商业环境中,产品生命周期管理(PLM)成为了企业提升竞争力、实现持续创新的关键。彩虹PLM作为业界领先的PLM解决方案,以其卓越的功能和创新的理念,引…

Python武器库开发-武器库篇之链接提取器(六十)

Python武器库开发-武器库篇之链接提取器(六十) 链接提取器介绍 链接提取器(Link Extractor)是一种用于从网页中提取链接的工具。它可以从网页的源代码中识别出所有的链接,并将这些链接提取出来。链接提取器可以用于各…

如何设置透明加密保护系统呢

设置透明加密保护系统通常涉及多个步骤,以下是一个基于参考文章信息和一般实践经验的清晰设置流程,包括分点表示和归纳: www.weaem.com 1. 需求分析 确定加密目标:明确需要加密的文件类型、存储位置和使用环境。评估安全风险&…

SQL Server 触发器

触发器是一种存储数据库过程,当数据库中发生特定事件(如插入、更新或删除操作)时,会自动调用该触发器。审核SQL Server实例的方法有很多,其中一种方法是使用审核触发器,触发器在SQL Server数据库中扮演着关…

AI发展面临的问题? —— AI对创造的重新定义

一、AI的问题描述 AI与数据安全问题:随着AI技术的发展和应用,数据安全问题日益突出。AI模型训练依赖于大量数据,而这些数据中可能包含个人隐私、商业秘密等敏感信息。如果数据在采集、存储、使用过程中处理不当,可能导致数据泄露或…