0

0

c# Saga模式和两阶段提交在c#中的实现

星降

星降

发布时间:2026-02-08 09:13:26

|

470人浏览过

|

来源于php中文网

原创

MassTransit Saga 是 C# 微服务中实现分布式事务补偿的成熟方案,通过状态机定义流转与 Compensate 方法声明补偿动作,需配合持久化、幂等性及 CorrelationId 使用。

c# saga模式和两阶段提交在c#中的实现

Saga 模式在 C# 中用 MassTransit 实现补偿逻辑

Saga 是处理长时间运行、跨服务事务的主流方案,C# 生态里最成熟的选择是 MassTransit —— 它原生支持基于状态机(StateMachineSaga)的 Saga 管理,自动持久化、幂等性、重试和补偿都内置了。

关键不是“手写事务日志”,而是定义状态流转和对应命令。比如订单创建后扣库存失败,必须触发“取消预留库存”操作,这个补偿动作由 Compensate 方法声明,不是靠 try-catch 手动调。

  • 必须实现 ISagaRepository,推荐用 EntityFrameworkSagaRepository 存到 SQL Server 或 PostgreSQL
  • 每个 Saga 实体需有唯一 CorrelationId(通常用 Guid),所有消息必须携带它,否则状态无法关联
  • 超时控制用 RequestTimeout + When(Timeout),别依赖外部定时器轮询
  • 不要在 When() 里直接调用 HTTP API;应发新命令(如 ReserveInventoryCommand),让下游消费者处理,保证解耦
public class OrderSaga : MassTransitStateMachine
{
    public State Submitted { get; private set; }
    public State InventoryReserved { get; private set; }
    public Event SubmitOrder { get; private set; }
    public Event InventoryReserved { get; private set; }
    public Event InventoryReservationFailed { get; private set; }
public OrderSaga()
{
    InstanceState(x =youjiankuohaophpcn x.CurrentState);

    Event(() =youjiankuohaophpcn SubmitOrder);
    Event(() =youjiankuohaophpcn InventoryReserved);
    Event(() =youjiankuohaophpcn InventoryReservationFailed);

    Initially(
        When(SubmitOrder)
            .Then(ctx =youjiankuohaophpcn ctx.Instance.OrderId = ctx.Data.OrderId)
            .TransitionTo(Submitted)
            .Send(context =youjiankuohaophpcn new ReserveInventoryCommand(context.Instance.OrderId)));

    During(Submitted,
        When(InventoryReserved)
            .TransitionTo(InventoryReserved),
        When(InventoryReservationFailed)
            .Call(ctx =youjiankuohaophpcn Console.WriteLine($"Rollback for {ctx.Instance.OrderId}"))
            .Compensate(ctx =youjiankuohaophpcn new CancelInventoryReservationCommand(ctx.Instance.OrderId)));
}

}

两阶段提交(2PC)在 C# 中不推荐直接实现

.NET 原生没有跨服务 2PC 支持,System.Transactions.TransactionScope 只适用于同进程内多个 SqlConnection 或支持 MSDTC 的资源,一旦涉及 HTTP、RabbitMQ、第三方 API,它就完全失效——不是功能限制,是协议层面不兼容。

常见误用:用 TransactionScope 包住 EF Core SaveChanges 和 HttpClient.PostAsync,以为能原子提交。实际结果是数据库改了,HTTP 请求失败,没人回滚数据库。

Android中JNI编程的那些事儿 中文WORD版
Android中JNI编程的那些事儿 中文WORD版

本文档主要讲述的是Android中JNI编程的那些事儿;JNI译为Java本地接口。它允许Java代码和其他语言编写的代码进行交互。在android中提供JNI的方式,让Java程序可以调用C语言程序。android中很多Java类都具有native接口,这些接口由本地实现,然后注册到系统中。希望本文档会给有需要的朋友带来帮助;感兴趣的朋友可以过来看看

下载
  • TransactionScope 要求所有参与者实现 IEnlistmentNotification,而 REST API、Kafka Producer、gRPC Client 都不实现它
  • 启用 MSDTC 在容器或云环境几乎不可行,且性能差、故障面大,超时默认 10 分钟,容易卡死资源
  • 即使本地多 DB 场景,EF Core 6+ 的 BeginTransactionAsync(isolationLevel) 也比 TransactionScope 更可控、无隐式分布式事务风险

什么时候该选 Saga,什么时候绕开分布式事务

核心判断依据是“业务是否允许中间态 + 补偿可行性”。例如电商下单:用户看到“已提交”,库存显示“已预留”,这是合法中间态;若支付失败,发补偿指令取消预留,整个流程可自愈。

  • 强一致性要求场景(如银行实时转账)→ 别用微服务拆,合并在单库单服务里用 ACID
  • 跨组织/第三方系统(如调微信支付、对接物流 SaaS)→ 只能靠 Saga + 对账,2PC 根本不存在
  • 高吞吐写入(如 IoT 设备上报)→ 用最终一致性 + 幂等写入,连 Saga 状态机都可能成为瓶颈,改用事件溯源 + 状态投影
  • 临时性数据(如购物车)→ 用 Redis + 过期时间,根本不需要事务语义

MassTransit Saga 的坑:持久化与幂等性没配对就等于没做

很多人只写状态机,却忘了配持久化,导致重启后 Saga 状态丢失,消息重复消费时无法识别“这单我已经处理过了”,直接双倍扣库存。

另一个高频问题是补偿消息没加幂等键。比如 CancelInventoryReservationCommand 被重发三次,库存就多加三次。正确做法是在命令里带 RequestId,消费者用该字段做去重(如存到 Redis Set 或 DB 唯一索引)。

  • EF Core 迁移必须包含 DbContext 中的 ISagaRepository 表(如 OrderState),否则启动报 InvalidOperation: No saga repository configured
  • 所有入站消息(SubmitOrder, InventoryReserved)必须设置 MessageIdCorrelationId,否则 ConsumeContext 拿不到上下文
  • 本地测试时禁用重试(UseInMemoryOutbox() + DisableRetry()),否则补偿逻辑会被干扰,难以验证状态流转

分布式事务从来不是“怎么实现”,而是“哪些地方根本不能分布”。Saga 不是银弹,但它是目前 C# 微服务里最靠谱的落地路径;硬上 2PC,多数时候只是把问题从代码移到运维日志里。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

857

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

331

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

351

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1468

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

365

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

1046

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

581

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

430

2024.04.29

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

39

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.4万人学习

Redis+MySQL数据库面试教程
Redis+MySQL数据库面试教程

共72课时 | 6.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号