0


消息队列RabbitMQ.03.死信交换机的讲解与使用

一、死信队列(延迟队列) 概念讲解


死信,在官网中对应的单词为“Dead Letter”,它是 RabbitMQ 的一种消息机制。

一般来说,生产者将消息投递到 broker 或者直接到 queue 里了,consumer 从 queue 取出消息进行消费,如果它一直无法消费某条数据,那么可以把这条消息放入死信队列里面。等待

条件满足了再从死信队列中取出来再次消费,从而避免消息丢失。

死信消息来源:

  • 消息 TTL 过期
  • 队列满了,无法再次添加数据
  • 消息被拒绝(reject 或 nack),并且 requeue =false

二、确认消息(局部方法处理消息)


默认情况下消息消费者是自动 ack (确认)消息的,如果要手动 ack(确认)则需要修改确认模式为 manual

spring:
  rabbitmq:
    listener:
      simple:
        acknowledge-mode: manual

三、代码实战


1.编写生产者代码,配置消息、直连交换机、路由键

 @Bean
    public Queue queueA() {//正常
        Map<String, Object> config = new HashMap<>();
        //message在该队列queue的存活时间最大为10秒
        config.put("x-message-ttl", 10000);
        //x-dead-letter-exchange参数是设置该队列的死信交换器(DLX)
        config.put("x-dead-letter-exchange", "ExchangeB");
        //x-dead-letter-routing-key参数是给这个DLX指定路由键
        config.put("x-dead-letter-routing-key", "bb");
        return new Queue("queueA",true,false,false,config);
    }

    @Bean
    public DirectExchange ExchangeA(){
        return new DirectExchange("ExchangeA");
    }

    @Bean
    public Binding bindingA(){
        return BindingBuilder
                .bind(queueA())
                .to(ExchangeA())
                .with("aa");
    }

    @Bean
    public Queue queueB() {
        return new Queue("queueB");
    }

    @Bean
    public DirectExchange ExchangeB(){
        return new DirectExchange("ExchangeB");
    }

    @Bean
    public Binding bindingB(){
        return BindingBuilder
                .bind(queueB())
                .to(ExchangeB())
                .with("bb");
    }
1.1代码解析:
  • Map<String, Object> config = new HashMap<>();
    //message在该队列queue的存活时间最大为10秒
    config.put("x-message-ttl", 10000);
    //x-dead-letter-exchange参数是设置该队列的死信交换器(DLX)
    config.put("x-dead-letter-exchange", "ExchangeB");
    //x-dead-letter-routing-key参数是给这个DLX指定路由键
    config.put("x-dead-letter-routing-key", "bb");
    return new Queue("queueA",true,false,false,config);*

首先我们需要知道死信产生的原因:

  • 消息过期: 消息在队列中等待的时间超过了指定的过期时间。
  • 消息被拒绝: 消费者拒绝消费消息,并且消息被标记为不可重新投递。
  • 队列满: 队列达到最大容量,无法再接收新的消息。

上述代码的作用是让消息等待10秒,如果10秒内没有进入消费者或者没有被操作那么它就会进入死信;我们就会将它放入消息B也就是当作死信就行处理。

return new Queue("queueA",true,false,false,config); 中参数的作用:

  • true: 持久化,表示队列在消息代理(例如 RabbitMQ)重启后仍然存在。
  • false: 非独占,表示该队列不会被其他连接独占使用。
  • false: 不自动删除,表示即使没有消费者连接,队列也不会被自动删除。
  • config: 包含了上述配置的 Map 对象,将这些配置应用到队列上。

总体来说,这段代码创建了一个具有消息过期和死信队列功能的队列 "queueA",并配置了过期消息发送到名为 "ExchangeB" 的交换器,并指定了死信的路由键为 "bb"。这样的配置在处理消息的时候能够更加灵活,并且对于消息的生命周期有了额外的控制。


2.配置消费者接受类接受直连交换机的路由键

queueA

@Component
@SuppressWarnings("all")
@Slf4j
@RabbitListener(queues = "queueA")
public class ReceiverQA {

    @RabbitHandler
    //手动确认消息
    public void process(String msg, Channel channel, @Header(AmqpHeaders.DELIVERY_TAG) long tag) throws Exception{
        log.warn("QA接收到:" + msg);
        //channel.basicAck(tag,true);
        //拒绝 ture是否重新入队 会一直 循环 false 会直接变成死信
        channel.basicReject(tag,true);
        Thread.sleep(1000);
    }
}
2.1. String msg​​​​​​​Channel channel​​​​​​​ @Header(AmqpHeaders.DELIVERY_TAG) ,long tag 方法参数解析:
  • Channel channel- Channel 对象表示与消息代理(例如 RabbitMQ)建立的通信通道。该通道提供了与消息代理进行交互的方法,如确认消息、拒绝消息等。在这个方法中,channel 参数用于与消息代理进行通信。
  • @Header(AmqpHeaders.DELIVERY_TAG) long tag- @Header 注解用于提取消息头中的信息。在这里,通过 AmqpHeaders.DELIVERY_TAG 提取了消息的传送标签(delivery tag)。- long tag 表示消息的唯一标识符。它是一个标识消息的数字,通常在消息代理传递消息给消费者时分配。这个标识符可以用于在确认、拒绝或重新排队消息时指定特定的消息。

总体来说,这个方法是一个消息处理方法,其中

message

参数表示消息的内容,

channel

参数用于与消息代理进行通信,而

tag

参数表示消息的唯一标识符,用于在消息确认、拒绝等操作中指定特定的消息。这样的方法通常用于处理从消息队列中接收到的消息。

*** 2.2.channel.basicAck(tag,true); 代码解析***

*** channel.basicAck(tag,true);***用于确认(acknowledge)消息的方法,通常在消费者成功处理消息后调用。以下是该方法的作用和参数含义:

  • channel- 这是表示与消息代理建立的通信通道的对象。在很多消息队列系统中,通常通过该通道进行消息的发布、消费等操作。
  • basicAck- 这是确认消息的方法。它告诉消息代理,消费者已经成功处理了某个特定的消息。
  • tag- 这是消息的唯一标识符或者句柄。在消费者接收到消息时,消息会被分配一个唯一的标识符,这个标识符用于确认或拒绝特定的消息。
  • true- 这是一个布尔值,通常表示确认多个消息。如果设置为 true,则表示确认到指定 tag 及其之前的所有消息。如果设置为 false,则仅确认指定的消息。
2.3.channel.basicReject(tag,true); 代码解析:

*** channel.basicReject(tag,true);***是在使用 AMQP(Advanced Message Queuing Protocol)中的通道(

channel

)对象时,用于拒绝一条消息的方法。让我们解释这个方法及其参数的意义:

  • tag 参数:- tag 表示消息的唯一标识符,通常是通过消费者获取的消息标签。它用于标识要拒绝的特定消息。
  • multiple 参数:- multiple 是一个布尔值,用于指定是拒绝单个消息还是多个消息。如果 multiplefalse,则表示仅拒绝标记为 tag 的单个消息。如果 multipletrue,则表示拒绝所有比 tag 小的、未被确认的消息。在这里,true 表示拒绝所有比 tag 小的未确认消息。
  • 作用:1. channel.basicReject(tag, true) 的作用是拒绝一条或多条消息。当消费者无法处理接收到的消息时,可以使用该方法将消息标记为拒绝,并根据需要将其重新排队或进入死信队列。2. 通过将 multiple 设置为 true,可以一次性拒绝多条消息,这对于批量处理消息的情况很有用。3. 拒绝消息会触发相应的处理机制,例如将消息重新排队或将其发送到死信交换器,具体取决于消息代理的配置。

总体来说,

channel.basicReject(tag, true)

是用于拒绝一条或多条消息的方法,其中

tag

表示消息的标签,而

multiple

决定了是拒绝单个消息还是多个消息。

**queueB **

@Component
@SuppressWarnings("all")
@Slf4j
@RabbitListener(queues = "queueB")
public class ReceiverQB {

    @RabbitHandler
    public void process(String msg) {
        log.warn("QB接收到:" + msg);
        //去数据库做修改
        //更改数据库的订单状态为取消
        //update order set status=-1 where id=#{id} and status=1
    }
}

3.对死信交换机进行测试

  @RequestMapping("/send6")
    public String send6(){
        template.convertAndSend("ExchangeA","aa","43249849234");
        return "🤣";
    }

最后页面网址访问方法**/send6**即可

标签: java java-rabbitmq

本文转载自: https://blog.csdn.net/d2916172682/article/details/135845616
版权归原作者 暴躁小段额 所有, 如有侵权,请联系我们删除。

“消息队列RabbitMQ.03.死信交换机的讲解与使用”的评论:

还没有评论