Spring Boot + RabbitMQ实现订单超时自动取消功能
场景:在京东下单,订单创建成功,等待支付,一般会给30分钟的时间,开始倒计时。如果在这段时间内
用户没有支付,则默认订单取消。
如何订单超时实现?
- 定时任务
- redission延时任务
- rabbitmq死信队列
本文将以rabbitmq死信队列展开做讲解,因为定时任务的方式,是有点问题的,原本业务系统希望10分钟后,如果订单未支付,就马上取消订单,并释放商品库存。但是一旦数据量大的话,就会加长获取未支付订单数据的时间,部分订单就做不到10分钟后取消了,可能是15分钟,20分钟之类的。这样的话,库存就无法及时得到释放,也就会影响成单数。而使用rabbitmq死信队列,在定义业务队列时可以考虑指定一个死信交换机,并绑定一个死信队列。当消息变成死信时,该消息就会被发送到该死信队列上,再取出订单信息进行判断订单是否已支付,如未支付则讲订单状态修改为取消状态,这样也是可以达到订单超时取消的需求的。
软件准备
erlang
请参考Win10下安装erlang
RabbitMQ
启动RabbitMQ,然后添加一个用户, 并给用户设置权限。
# 后台启动
rabbitmq-server -detached
# 添加用户
rabbitmqctl add_user root 123456
# 设置用户权限:
rabbitmqctl set_permissions -p "/" root ".*" ".*" ".*"
接下来创建一个springboot工程,并集成RabbitMQ。
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.2.8.RELEASE</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
<groupId>com.lagou</groupId>
<artifactId>rabbitmq-work</artifactId>
<version>0.0.1-SNAPSHOT</version>
<name>rabbitmq-work</name>
<description>Demo project for Spring Boot</description>
<properties>
<java.version>1.8</java.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.47</version>
<scope>runtime</scope>
</dependency>
<!--mybatis-plus依赖-->
<dependency>
<groupId>com.baomidou</groupId>
<artifactId>mybatis-plus-boot-starter</artifactId>
<version>3.3.2</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
接下来在application.yml配置文件中假如rabbitmq配置
spring:
application:
name: rabbit-work
# rabbitmq配置
rabbitmq:
host: localhost
virtual-host: /
username: root
password: 123456
# redis配置
redis:
timeout: 6000
host: localhost
port: 6379
database: 0
# 数据源配置
datasource:
url: jdbc:mysql://localhost:3306/order?useUnicode=true&characterEncoding=utf8&useSSL=false
driver-class-name: com.mysql.jdbc.Driver
username: root
password: 123456
hikari:
minimum-idle: 3
maximum-pool-size: 5
max-lifetime: 30000
connection-init-sql: SELECT 1
# mybatis-plus配置
mybatis-plus:
mapper-locations: classpath:com.lagou.*.mapper/*.xml
type-aliases-package: com.lagou.*.domain
定义RabbitConfig
@Configuration
public class RabbitConfig {
/**
* 订单队列
*
* @return {@link Queue}
*/
@Bean
public Queue orderQueue() {
Map<String, Object> argments = new HashMap<>();
argments.put("x-message-ttl", 60000);
argments.put("x-dead-letter-exchange", RabbitConstants.ORDER_DLX_EXCHANGE);
argments.put("x-dead-letter-routing-key", RabbitConstants.ORDER_DLX_ROUTING_KEY);
Queue queue = new Queue(RabbitConstants.ORDER_QUEUE, true, false, false, argments);
return queue;
}
/**
* 订单交换机
*
* @return {@link Exchange}
*/
@Bean
public Exchange orderExchange() {
return new DirectExchange(RabbitConstants.ORDER_EXCHANGE, true, false, null);
}
/**
* 订单路由键
*
* @return {@link Binding}
*/
@Bean
public Binding orderRouting() {
return BindingBuilder.bind(orderQueue()).to(orderExchange()).with(RabbitConstants.ORDER_ROUTING_KEY).noargs();
}
/**
* 订单死信队列
*
* @return {@link Queue}
*/
@Bean
public Queue orderDlxQueue() {
Queue queue = new Queue(RabbitConstants.ORDER_DLX_QUEUE, true, false, false);
return queue;
}
/**
* 订单死信交换机
*
* @return {@link Exchange}
*/
@Bean
public Exchange orderDlxExchange() {
return new DirectExchange(RabbitConstants.ORDER_DLX_EXCHANGE, true, false, null);
}
/**
* 订单死信路由键
*
* @return {@link Binding}
*/
@Bean
public Binding orderDlxRouting() {
return BindingBuilder.bind(orderDlxQueue()).to(orderDlxExchange()).with(RabbitConstants.ORDER_DLX_ROUTING_KEY).noargs();
}
/**
* 库存队列
*
* @return {@link Queue}
*/
@Bean
public Queue stockQueue() {
return new Queue(RabbitConstants.STOCK_QUEUE, true, false, false, null);
}
/**
* 库存交换机
*
* @return {@link Exchange}
*/
@Bean
public Exchange stockExchange() {
return new DirectExchange(RabbitConstants.STOCK_EXCHANGE, true, false, null);
}
/**
* 库存路由键
*
* @return {@link Binding}
*/
@Bean
public Binding stockRouting() {
return BindingBuilder.bind(stockQueue()).to(stockExchange()).with(RabbitConstants.STOCK_ROUTING_KEY).noargs();
}
}
实现消息发送
@RequestMapping(value = "/submit", produces = "application/json;charset=UTF-8")
public R submit(@RequestBody OrderVo orderVo) throws UnsupportedEncodingException {
Order order = orderService.createOrder(orderVo);
System.out.println("OrderController.submit... createOrder");
// 放入死信队列
amqpTemplate.convertAndSend(RabbitConstants.ORDER_EXCHANGE,
RabbitConstants.ORDER_ROUTING_KEY,
(order.getId() + ""));
System.out.println("OrderController.submit... sendMessage to orderExchange");
return R.ok(order);
}
消息消费者监听
@Component
public class OrderHandler {
@Autowired
private OrderService orderService;
@RabbitListener(queues = RabbitConstants.ORDER_DLX_QUEUE, ackMode = "MANUAL")
public void onMessage(Message message, Channel channel) throws IOException {
System.out.println("消息进入死信队列...");
String s = new String(message.getBody());
Order o = orderService.getById(Long.parseLong(s));
if (o != null && OrderConstants.TOPAID.getCode().equals(o.getOrderState())) {
Order order = new Order();
order.setId(o.getId());
order.setOrderState(OrderConstants.CANCEL.getCode());
order.setGmtModified(new Date());
orderService.updateById(order);
}
// 手动ack
channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);
}
}
启动Spring Boot应用进行测试
点击其中一个有库存的商品进行购买,先演示一个成功支付的。
接下来再演示支付失败的,并注意控制台日志打印。
通过监听死信队列,消息在进入死信队列之后就可以做一系列业务逻辑处理,比如,消息如果还是未支付状态,将其修改为取消支付。
__EOF__
作 者:Jerry
出 处:https://www.cnblogs.com/jerry0612/p/14592389.html
关于博主:编程路上的小学生,热爱技术,喜欢专研。评论和私信会在第一时间回复。或者直接私信我。
版权声明:署名 - 非商业性使用 - 禁止演绎,协议普通文本 | 协议法律文本。
声援博主:如果您觉得文章对您有帮助,可以点击文章右下角【推荐】一下。您的鼓励是博主的最大动力!
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!