使用 RabbitMQ 实现秒杀订单系统的异步消息处理

news/2025/1/23 6:51:16/

使用 RabbitMQ 实现秒杀订单系统的异步消息处理

在秒杀系统中,如何确保高并发环境下的订单处理稳定高效是个很大的挑战。为了解决这个问题,我们通常会引入消息队列,通过异步处理来削峰填谷。这篇文章将详细讲解如何使用 RabbitMQ 来设计一个秒杀订单系统的异步消息处理流程,重点是如何使用交换机(Exchange)、队列(Queue)、路由键(Routing Key)以及死信队列(Dead Letter Queue)来管理订单的状态。

1. 为什么需要使用消息队列?

在秒杀系统中,用户的请求量在极短时间内会成倍增加。如果每个请求都直接访问数据库,极有可能导致数据库崩溃甚至出现库存超卖的情况。为了避免这种情况,通常会通过引入 RabbitMQ 来将订单处理变为异步,先把订单请求写入队列,然后由消费者从队列中逐个取出进行处理。这样一来,前端响应速度会加快,同时也减轻了数据库的压力。

2. 系统设计概述

在这套秒杀系统中,我们设计了以下几个关键组件:

  • 秒杀订单的主交换机和队列:用于处理异步下单请求。
  • 成功和失败的订单队列:分别处理订单处理成功和失败的消息。
  • 死信队列(Dead Letter Queue):处理超时或者其他异常导致的消息失败。

我们使用了 TopicExchange 来根据路由键将消息发送到不同的队列。每个订单请求在被处理时,首先会进入秒杀队列,然后根据处理结果被转发到成功或失败的队列。处理成功的订单如果超时未支付,会被转发到死信队列,进行进一步处理(例如取消订单、回滚库存等)。

3. 配置 RabbitMQ

下面我们通过配置类 RabbitMQConfig 来详细展示如何定义和绑定 RabbitMQ 的交换机、队列、路由键,以及死信队列。

3.1 定义交换机

在 RabbitMQ 中,交换机是负责将消息路由到相应队列的组件。我们定义了三个交换机:

  • SECKILL_EXCHANGE:用于秒杀队列的交换机,负责接收用户的下单请求。
  • SECKILL_EXCHANGE_ORDER:用于订单处理的交换机,将订单路由到成功或失败的队列。
  • DEAD_LETTER_EXCHANGE_SECKILL_ORDER:死信交换机,处理那些订单失败或者超时的情况。
@Bean
public TopicExchange seckillExchange() {return new TopicExchange(SECKILL_EXCHANGE);
}@Bean
public TopicExchange seckillExchangeOrder() {return new TopicExchange(SECKILL_EXCHANGE_ORDER);
}@Bean
public TopicExchange deadLetterExchangeOrder(){return new TopicExchange(DEAD_LETTER_EXCHANGE_SECKILL_ORDER);
}

通过以上代码,我们分别为秒杀订单、成功/失败处理、以及死信处理定义了各自的交换机。

3.2 定义队列

队列是存放消息的地方。我们定义了以下几个队列:

  • SECKILL_QUEUE:秒杀队列,用户下单请求会被异步地写入这个队列。
  • SECKILL_QUEUE_ORDER_SUCCESS:处理成功订单的队列,订单处理完成后消息会进入这里。
  • SECKILL_QUEUE_ORDER_FAIL:处理失败订单的队列。
  • DEAD_LETTER_QUEUE_SECKILL_ORDER:死信队列,用于处理超时或者异常未完成的订单。
@Bean
public Queue seckillQueue() {return new Queue(SECKILL_QUEUE, true);
}@Bean
public Queue seckillQueueOrderSuccess() {return QueueBuilder.durable(SECKILL_QUEUE_ORDER_SUCCESS).withArgument("x-dead-letter-exchange", DEAD_LETTER_EXCHANGE_SECKILL_ORDER).withArgument("x-dead-letter-routing-key", DEAD_LETTER_ROUTINGKEY_SECKILL_ORDER).withArgument("x-message-ttl", 300000)  // 消息5分钟过期.build();
}@Bean
public Queue seckillQueueOrderFail() {return new Queue(SECKILL_QUEUE_ORDER_FAIL, true);
}@Bean
public Queue deadLetterQueueSeckillOrder() {return new Queue(DEAD_LETTER_QUEUE_SECKILL_ORDER, true);
}

其中,seckillQueueOrderSuccess 队列设置了一个 5 分钟的 TTL(消息过期时间)。如果订单在 5 分钟内未支付,消息将被发送到死信队列,以便进行进一步处理。

3.3 绑定队列和交换机

RabbitMQ 的消息流转需要通过绑定(Binding)来实现。绑定将队列和交换机通过路由键连接在一起。我们为每个队列和交换机进行了相应的绑定:

// 秒杀队列与秒杀交换机绑定
@Bean
public Binding seckillBinding(Queue seckillQueue, TopicExchange seckillExchange) {return BindingBuilder.bind(seckillQueue).to(seckillExchange).with(SECKILL_ROUTINGKEY);
}// 秒杀成功订单队列和秒杀订单交换机绑定
@Bean
public Binding seckillOrderSuccessBinding(Queue seckillQueueOrderSuccess, TopicExchange seckillExchangeOrder){return BindingBuilder.bind(seckillQueueOrderSuccess).to(seckillExchangeOrder).with(SECKILL_ROUTINGKEY_ORDER_SUCCESS);
}// 秒杀失败订单队列和秒杀订单交换机绑定
@Bean
public Binding seckillOrderFailBinding(Queue seckillQueueOrderFail, TopicExchange seckillExchangeOrder){return BindingBuilder.bind(seckillQueueOrderFail).to(seckillExchangeOrder).with(SECKILL_ROUTINGKEY_ORDER_FAIL);
}// 绑定死信队列到死信交换机
@Bean
public Binding bindingDeadLetterQueue(Queue deadLetterQueueSeckillOrder, TopicExchange deadLetterExchangeOrder) {return BindingBuilder.bind(deadLetterQueueSeckillOrder).to(deadLetterExchangeOrder).with(DEAD_LETTER_ROUTINGKEY_SECKILL_ORDER);
}

通过这些绑定,消息会根据路由键从交换机流向相应的队列。比如用户的订单消息会先进入 SECKILL_QUEUE,然后根据订单的处理结果转发到 SECKILL_QUEUE_ORDER_SUCCESSSECKILL_QUEUE_ORDER_FAIL

4. 死信队列的作用

在这套设计中,死信队列的作用至关重要。死信队列用于接收那些无法正常处理的消息,比如订单支付超时或者系统异常。死信队列绑定了一个死信交换机,当消息过期(如上文的 5 分钟 TTL)或者被拒绝时,会自动进入死信队列。这样我们就能针对这些问题订单进行统一处理,比如取消订单、回滚库存等操作。

文章来源:https://blog.csdn.net/cxkjntm123/article/details/142099414
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.ppmy.cn/news/1525653.html

相关文章

Linux通过特定端口查看服务是否启动

Linux通过特定端口查看服务是否启动 你可以使用netstat或ss命令来检查特定端口上的服务。例如&#xff0c;使用ss -tuln | grep <端口号>来查看端口是否被占用。 netstat 你可以使用以下命令来查看特定端口上的服务&#xff1a; netstat -tuln | grep <端口号>…

VPP -LB 命令配置

【组网拓扑】 ping --> 2 1.1.1.3 【1.1.1.1 lb 2.2.2.2】 - 1.1.1.2 - 1.1.1.4 【GRE方式配置】 set interface state GigabitEthernet0/8/0 up set interface ip address GigabitEthernet0/8/0 1.1.1.1/24 lb conf ip4-src-addr…

看Threejs好玩示例,学习创新与技术(二)

本文接上篇内容&#xff0c;继续挖掘应用ThreeJS的一些创新算法。 本文理解难度比较大&#xff0c;可以先看一些概念&#xff0c;在难的地方培养一些意识即可。 1、扭曲的自然 下面图本身是矩形的&#xff0c;为何它可以这么扭曲呢&#xff1f;它在随机处带有一定的规律&…

Android - NDK:在Jni中打印Log信息

在Jni中打印Log信息 1、在配置CMakeLists.txt find_library( # Sets the name of the path variable.log-lib# Specifies the name of the NDK library that# you want CMake to locate.log)# Specifies libraries CMake should link to your target library. You # can link…

laravel 资源show方法问题

有张admin_users表&#xff0c;但是在控制器的show方法返回的数据时空。 路由 Route::resource(adminuser, \App\Http\Controllers\AdminUserController::class)->except([create, edit]); 解决&#xff1a; 把路由adminuser 改成 admin-user Route::resource(admin-user, …

深度学习:怎么看pth文件的参数

.pth 文件是 PyTorch 模型的权重文件&#xff0c;它通常包含了训练好的模型的参数。要查看或使用这个文件&#xff0c;你可以按照以下步骤操作&#xff1a; 1. 确保你有模型的定义 你需要有创建这个 .pth 文件时所用的模型的代码。这意味着你需要有模型的类定义和架构。 2. …

【刷题】Day4--密码检查

Hi&#xff01; 今日刷题&#xff0c;小白一枚&#xff0c;欢迎指导 ~ 【链接】 密码检查_牛客题霸_牛客网 【思路】 依次根据规则判断密码是否合格。while里嵌套个for循环&#xff0c;来进行密码的多组输入&#xff0c;for循环进行一次代表判断一个密码串&#xff1b;规则…

灌区信息化发展趋势展望

灌区信息化作为现代农业发展的重要组成部分&#xff0c;正逐渐成为提升水资源管理效率、保障粮食安全与促进农业可持续发展的关键途径。随着信息技术的飞速进步和智能化技术的广泛应用&#xff0c;灌区信息化的未来发展趋势展现出多维度、深层次的变革与创新&#xff0c;其发展…