秒杀-订单创建消费者CreateOrderConsumer

这个类是基于MassTransit框架的消息消费者,负责接收 "库存扣减成功" 的消息并创建秒杀订单,同时通过消息通知库存服务确认订单状态。

复制代码
using MassTransit;
using MassTransit.Transports;
using SqlSugar;
using System.Transactions;
using ZR.Common.Model;
using ZR.Model.Stock;
using ZR.Service.Stock.IStockService;

/// <summary>
/// 订单创建消息消费者
/// 负责消费 CreateOrderMessage 消息,完成秒杀订单的创建
/// </summary>
public class CreateOrderConsumer : IConsumer<CreateOrderMessage>
{
    // 本地消息表服务(操作库存服务的消息记录)
    private readonly IStockMessageService _StockMessageService;
    // 秒杀订单服务(操作订单数据库)
    private readonly ISeckillOrderService _SeckillOrderService;
    // SqlSugar 数据库客户端(用于事务管理)
    private readonly ISqlSugarClient _SqlSugarClient;
    // MassTransit 消息总线(用于发送确认消息)
    private readonly IBus _bus;

    /// <summary>
    /// 构造函数注入依赖
    /// 通过依赖注入获取所需服务,解耦组件间依赖
    /// </summary>
    public CreateOrderConsumer(IStockMessageService stockMessageService, ISeckillOrderService seckillOrderService,
                                ISqlSugarClient sqlSugarClient, IBus bus)
    {
        _StockMessageService = stockMessageService;
        _SeckillOrderService = seckillOrderService;
        _SqlSugarClient = sqlSugarClient;
        _bus = bus;
    }

    /// <summary>
    /// 消费消息的核心方法
    /// 当 CreateOrderMessage 消息到达时自动调用
    /// </summary>
    public async Task Consume(ConsumeContext<CreateOrderMessage> context)
    {
        Console.WriteLine("_____________________CreateOrderConsumer____________");
        var message = context.Message; // 获取消息内容(订单号、商品ID、用户ID等)

        // 1. 幂等性检查:判断订单是否已存在(防止消息重复消费导致重复创建订单)
        var exist = await _SeckillOrderService.GetFirstAsync(x => x.OrderNo == message.OrderNo);
        if (exist != null)
        {
            // 订单已存在:直接发送"订单创建确认"消息,通知库存服务更新消息状态
            var uri = new Uri("exchange:stockConfirmQueue.direct?type=direct"); // 目标交换机地址
            var endPonit = await _bus.GetSendEndpoint(uri); // 获取消息发送端点
            if (endPonit == null)
                throw new Exception("消息创建失败");

            // 发送确认消息(包含订单号)
            await endPonit.Send(new ConfirmOrderCreatedMessage
            {
                OrderNo = message.OrderNo,
            }, ctx =>
            {
                ctx.SetRoutingKey("orderpublic"); // 指定路由键,匹配库存确认队列
            });
            return;
        }

        // 2. 订单不存在:创建新订单
        bool dbok = false; // 数据库事务是否成功
        string mess = "";  // 事务信息

        try
        {
            // 使用 SqlSugar 事务包装订单创建操作(确保原子性)
            (dbok, mess) = await _SqlSugarClient.UseTranAsync(async () =>
            {
                // 创建秒杀订单实体
                var order = new SeckillOrder
                {
                    OrderNo = message.OrderNo,   // 订单号(与库存服务生成的一致)
                    GoodsId = message.GoodsId,   // 商品ID
                    UserId = message.UserId,     // 用户ID
                    Status = "0",                // 订单状态(0-待支付)
                    CreateTime = DateTime.Now,   // 创建时间
                };
                // 保存订单到数据库
                _SeckillOrderService.Add(order);
            });

            // 3. 订单创建成功后,发送"订单创建确认"消息
            var uri = new Uri("exchange:stockConfirmQueue.direct?type=direct");
            var endPonit = await _bus.GetSendEndpoint(uri);
            if (endPonit == null)
                throw new Exception("消息创建失败");

            await endPonit.Send(new ConfirmOrderCreatedMessage
            {
                OrderNo = message.OrderNo,
            }, ctx =>
            {
                ctx.SetRoutingKey("orderpublic");
            });
        }
        catch (Exception ex)
        {
            // 异常时直接返回(依赖 MassTransit 重试机制重新消费消息)
            return;
        }
    }
}
相关推荐
马克Markorg1 小时前
常见的向量数据库和具有向量数据库能力的数据库
数据库
郝学胜-神的一滴2 小时前
深入解析Python字典的继承关系:从abc模块看设计之美
网络·数据结构·python·程序人生
绵绵细雨中的乡音2 小时前
深入理解 ET 与 LT 模式及其在 Reactor 模型中的应用
服务器·网络·php
暖馒3 小时前
Modbus应用层协议的深度剖析
网络·网络协议·c#·wpf·智能硬件
Coder_Boy_4 小时前
技术让开发更轻松的底层矛盾
java·大数据·数据库·人工智能·深度学习
helloworldandy4 小时前
使用Pandas进行数据分析:从数据清洗到可视化
jvm·数据库·python
yunfuuwqi5 小时前
OpenClaw✅真·喂饭级教程:2026年OpenClaw(原Moltbot)一键部署+接入飞书最佳实践
运维·服务器·网络·人工智能·飞书·京东云
迎仔5 小时前
C-算力中心网络隔离实施方法:怎么搞?
运维·网络
代码游侠5 小时前
C语言核心概念复习——网络协议与TCP/IP
linux·运维·服务器·网络·算法
数据知道6 小时前
PostgreSQL 故障排查:如何找出数据库中最耗时的 SQL 语句
数据库·sql·postgresql