RabbitMQ之延迟消息实战

RabbitMQ之延迟消息实战

在这里插入图片描述

  1. 使用死信交换机实现延迟消息
    在这里插入图片描述
    使用死信交换机的过期时间以及没有消费者进行消费,时间到了就会到死信队列中,由此可以实现延迟消息
  2. 使用延迟消息插件
    前提:需要mq配置插件
    在这里插入图片描述
    在这里插入图片描述
  3. 延时信息案例实战
    把一个30分钟的延迟消息可以拆分成各个小时间段来优化MQ大量信息的压力
    在这里插入图片描述
    在这里插入图片描述
    代码实现
    (1)定义消息体对象
@Data
public class MultyDelayMessage<T>{
	// 消息体
	private T data;
	// 记录延迟时间的集合
	private List<Long> delayMillis;
	
	public MultiDelayMessage(T data List<Long> delayMilis){
		this.data = data;
		this.delayMillis = delayMillis;
	}
	public static <T> MultiDelayMessage<T> of(T data,Long ... delayMillis){
		return new MultiDelayMessage<>(data,Collutils.newArrayList(delayMilis));
	}
	//获取并移除下一个延迟时间,返回队列中的第一个延时时间
	public Long removeNextDelay{
		return delayMillis.remove(0);
	}
	// 判断是否有下一个延时时间
	public bollean hasNextDelay(){
		return !delayMilis.isEmpty();
	}
}

(2)定义队列名称

public interface MqConstants{
	String DELAY_EXCHANGE = "trade.delay.topic";
	String DELAY_ORDER_QUEUE = "trade.order.delay.queue";
	String DELAY_ORDER_ROUTING_KEY = "order.query"-;
}

(3)创建订单发送延迟消息

MultiDelayMessage<Long> msg = MultiDelayMessage.of(order.getId(),10000L,10000L,10000L,15000L,15000L,30000L,30000L);
rabbitTemplate.convertAndSend(MqConstants,DELAY_EXCHANGE,DELAY_ORDER_ROUTING_KEY,msg,new MessagePostProcess{
	@Override
	public Message postProcessMessage(Message message) throw AmqpException{
		message.getMessageProperties().setDelay(msg.removeNextDelay().intValue());
	}
});

(4)监听到延迟消息处理订单状态

@Component
public class OrderStatusChechListenter{
	
	@Autowrid
	private OrderService orderService;

	@RabbitListener(@QueueBinding(value=@Queue(value=MqConstants.DELAY_ORDER_QUEUE,durable="true"),exchange=@Exchange(value=MqConstants.DELAY_EXCHANGE,delayed="true",type=ExchangeTypes.TOPIC),key=MqConstants.DELAY_ORDER_ROUTING_KEY))
public void listenOrderDelayMessage(MultiDelayMessage message){
	// 查询订单支持状态
	Order order = orderService.getById(message.getData());
	if(order == null || order.getStatus == 2){// 已支付
		return;
	}
	// 查询支付服务该订单的支付状态
	boolean isPay = payServive.getById(order.getBussesesId());
	if(isPay){
		// 已支付
		orderService.markOrderPaySuccess(order.getId());
		return;
	}
	// 未支付,并且有延迟时间
	if(msg.hasNextDelay()){
		//重发延迟消息
		Long nextDelay = msg.removeNextDelay();
		rabbitTemplate.convertAndSend(MqConstants,DELAY_EXCHANGE,DELAY_ORDER_ROUTING_KEY,msg,new MessagePostProcess{
	@Override
	public Message postProcessMessage(Message message) throw AmqpException{
		message.getMessageProperties().setDelay(nextDelay.intValue());	}
});
	return;
	}
	// 没有延迟消息,取消订单
	orderService.cancleOrder(order.getId());
	//恢复库存
	orderService.updateStore(order.getId());
}

}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
延迟队列在需要延时处理消息的场景下非常有用。使用RabbitMQ来实现延迟队列可以充分利用RabbitMQ的特性,如可靠发送和投递消息,以及死信队列来保证消息至少被消费一次,未被处理的消息也不会被丢弃。通过RabbitMQ集群的特性,可以解决单点故障的问题,即使某个节点挂掉,延迟队列仍然可用且消息不会丢失。\[1\] 在实现延迟队列的过程中,可以选择多种方式。比如利用Java的DelayQueue、Redis的zset、Quartz(定时器)或者Kafka的时间轮等。每种方式都有其适用的场景,需要根据具体需求来选择。\[1\] 关于RabbitMQ延迟队列实战,可以通过以下步骤来实现: 1. 安装Windows版Docker和RabbitMQ。 2. 使用Maven整合RabbitMQ,实现消息的生产和消费。 3. 使用RabbitMQ的发布确认和交换机确认来确保消息的可靠性。 4. 使用RabbitMQ消息回报(队列确认)来处理未被消费的消息。 5. 可以考虑使用RabbitMQ的备份交换机和优先级队列来进一步增强延迟队列的功能。\[2\] 具体实现方式可以使用RabbitMQ提供的死信路由机制。当一个消息的时间戳到期时,如果还没有被消费,则会被转发到死信路由,消费者可以绑定到这个死信路由上来处理延迟消息。\[3\] #### 引用[.reference_title] - *1* [消息中间件 RabbitMQ延迟队列 详解&实战](https://blog.csdn.net/qq_52567278/article/details/124427476)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [RabbitMQ延迟队列及实战](https://blog.csdn.net/m0_68681879/article/details/129547212)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [【项目实战篇】基于 RabbitMQ 实现延迟队列](https://blog.csdn.net/qq_18244417/article/details/117050723)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值