当前位置: 首页 > article >正文

RabbitMQ延时队列

RabbitMQ延时队列

什么是延时队列

延时队列顾名思义,即放置在该队列里面的消息是不需要立即消费的,而是等待一段时间之后取出消费。

应用场景

场景一:在订单系统中,一个用户下单之后通常有30分钟的时间进行支付,如果30分钟之内没有支付成功,那么这个订单将进行一场处理。这是就可以使用延时队列将订单信息发送到延时队列。

场景二:用户希望通过手机远程遥控家里的智能设备在指定的时间进行工作。这时候就可以将用户指令发送到延时队列,当指令设定的时间到了再将指令推送到智能设备。

实现方式

Rabbitmq实现延时队列一般而言有两种形式: 第一种方式:利用两个特性: Time To Live(TTL消息过期)、Dead Letter Exchanges(DLX死信队列) 第二种方式:利用rabbitmq中的插件x-delay-message

三、第一种:利用TTL DLX实现延时队列的方式 AMQP协议和RabbitMQ队列本身没有直接支持延迟队列功能,但是可以通过以下特性模拟出延迟队列的功能。

1、Time To Live(TTL)

RabbitMQ可以针对Queue设置x-expires 或者 针对Message设置 x-message-ttl,来控制消息的生存时间,如果超时(两者同时设置以最先到期的时间为准),则消息变为dead letter(死信)

A: 通过队列属性设置,队列中所有消息都有相同的过期时间。 B: 对消息进行单独设置,每条消息TTL可以不同。

2、Dead Letter Exchanges(DLX)

RabbitMQ的Queue可以配置x-dead-letter-exchange和x-dead-letter-routing-key(可选)两个参数,如果队列内出现了dead letter,则按照这两个参数重新路由转发到指定的队列。 x-dead-letter-exchange:出现dead letter之后将dead letter重新发送到指定exchange x-dead-letter-routing-key:出现dead letter之后将dead letter重新按照指定的routing-key发送

用一个具体案例来实现第一种方式:用户下订单后,如何在一分钟没有支付就取消订单

死信队列

死信的概念

死信,顾名思义就是无法被消费的消息,字面意思可以这样理解,一般来说,producer 将消息投递到 broker 或者直接到queue 里了,consumer 从 queue 取出消息进行消费,但某些时候由于特定的原因导致 queue 中的某些消息无法被消费,这样的消息如果没有后续的处理,就变成了死信,有死信自然就有了死信队列。

应用场景:为了保证订单业务的消息数据不丢失,需要使用到 RabbitMQ 的死信队列机制,当消息消费发生异常时,将消息投入死信队列中.还有比如说: 用户在商城下单成功并点击去支付后在指定时间未支付时自动失效

死信的来源

  • 消息 TTL 过期

  • 队列达到最大长度(队列满了,无法再添加数据到 mq 中)

  • 消息被拒绝(basic.reject 或 basic.nack)并且 requeue=false.

死信实战

package com.zking.xiongda.config;

import org.springframework.amqp.core.*;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.stereotype.Component;

import java.util.HashMap;
import java.util.Objects;

@Configuration
public class QueueConfig {
    /**
     * 死信队列
     * @return
     */
    @Bean
    public Queue deadQueue(){
        return new Queue("dead-queue");
    }

    /**
     * 死信交互机
     */
    @Bean
    public Exchange deadExchange(){
        return new DirectExchange("dead-exchange");
    }

    /**
     * 绑定死信队列
     * @return
     */
    @Bean
    public Binding bindingExchange(){
        return BindingBuilder.bind(deadQueue())
                .to(deadExchange())
                .with("dead-exchange.key")
                .noargs();
    }


    /**
     * 普通队列
     * @return
     */
    @Bean
    public Queue xioangdaQueue(){
        // hashmap设置对列参数信息
        HashMap<String, Object> hashMap = new HashMap<>();
        // 设置消息过期时间
        hashMap.put("x-message-ttl",20000);
        // 设置消息过期转换死信队列交换机
        hashMap.put("x-dead-letter-exchange","dead-exchange");
        // 设置改队列关联转移到交换机路由key
        hashMap.put("x-dead-letter-routing-key","dead-exchange.key");

        // true是否持久,false:是否排他,false:是否自动删除
        return new Queue("xiongda-queue",true,false,false,hashMap);
    }

    /**
     * 普通交换机
     */
    @Bean
    public Exchange xioangdaExchange(){
        return new DirectExchange("xiongda-exchange");
    }

    /**
     * 绑定普通队列
     * @return
     */
    @Bean
    public Binding xioangdaDindingExchange(){
        return BindingBuilder.bind(xioangdaQueue())
                .to(xioangdaExchange())
                .with("xiongda-exchange.key")
                .noargs();
    }




}


http://www.kler.cn/a/427666.html

相关文章:

  • 移植NIOS10.1工程,NIOS10.1路径修改
  • Java 8 Stream API 新特性详解及实践
  • 63 基于单片机的四个速度比较
  • PMP–一、二、三模、冲刺–分类–10.沟通管理
  • 基于微信小程序的教学质量评价系统
  • mysql 1主2从搭建
  • v-on:click后面可以加什么
  • 并查集—河工oj
  • 利用PHP和GD库实现图片切割
  • 项目五 李白个人生平
  • 可供参考的GitHub国内镜像
  • DApp浏览器能否集成在自己开发的DApp里?
  • 安全审计系统
  • Mac如何安装SVN
  • 扫描IP段内的使用的IP
  • 经典的CNN架构
  • 在玩“吃鸡”的时候游戏崩溃要如何解决?游戏运行时崩溃是什么原因?
  • k8s-Informer之Reflector的解析
  • 在Node.js局域网调试https的Vue项目
  • el-select的搜索功能