RabbitMq之死信队列

   DLX ,全称为 Dead-Letter-Exchange ,可以称之为死信交换器,也有人称之为死信邮箱。当消息在一个队列中变成死信 (dead message) 之后,它能被重新被发送到另一个交换器中,这个交换器就是 DLX,绑定 DLX 的队列就称之为死信队列。消息变成死信一般是由于以下几种情况:

  1. 1.消息被拒绝 (Basic.Reject/Basic .Na ck) ,井且设置 requeue 参数为 false;
  2. 2.消息过期;
  3. 3.队列达到最大长度。

  DLX 也是一个正常的交换器,和一般的交换器没有区别,它能在任何的队列上被指定 , 实际上就是设置某个队列的属性。当这个队列中存在死信时 , RabbitMQ 就会自动地将这个消息重新发布到设置的 DLX 上去 ,进而被路由到另一个队列,即死信队列。可以监听这个队列中的消息、以进行相应的处理,这个特性与将消息的 TTL 设置为 0 配合使用可以弥补 imrnediate 参数的功能。
 

/**
 * Author: dyh
 * Date:   2019/4/26
 * Description:
 * 生产者首先发送一条携带路由键为 " rk " 的消息,然后经过交换器
 exchange .normal 顺利地存储到队列 queue.normal 中 。由于队列 queue.normal 设置了过期时间为
 10s , 在这 10s 内没有消费者消费这条消息,那么判定这条消息为过期。由于设置了 DLX , 过期
 之时 , 消息被丢给交换器 exchange.dlx 中,这时找到与 exchange.dlx 匹配的队列 queue .dlx , 最
 后消息被存储在 queue.dlx 这个死信队列中。
 */
public class Send
{
    public static void main(String[] args) throws IOException,TimeoutException {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();
        //声明2个交换器,分别绑定2个队列
        channel.exchangeDeclare("exchange.dlx","direct",true);
        channel.exchangeDeclare("exchange.normal","fanout",true);
        HashMap map = new HashMap<>();
        map.put("x-dead-letter-exchange","dlx_exchange");
        map.put("x-dead-letter-routing-key","routingkey");
        map.put("x-message-ttl",10000);
        channel.queueDeclare("queue.normal", true, false, false, map);
        channel.queueBind("queue.normal","exchange.normal","");
        channel.queueDeclare("queue.dlx", true, false, false, null);
        channel.queueBind("queue.dlx","exchange.dlx","routingkey");

        channel.basicPublish("exchange.normal", "rk", MessageProperties.PERSISTENT_TEXT_PLAIN, "dlx".getBytes());

        channel.close();
        connection.close();
    }
}

生产者首先发送一条携带路由键为 " rk " 的消息,然后经过交换器exchange .normal 顺利地存储到队列 queue.normal 中 。由于队列 queue.normal 设置了过期时间为10s , 在这 10s 内没有消费者消费这条消息,那么判定这条消息为过期。由于设置了 DLX , 过期之时 , 消息被丢给交换器 exchange.dlx 中,这时找到与 exchange.dlx 匹配的队列 queue .dlx , 最
后消息被存储在 queue.dlx 这个死信队列中。

运行后观察web页面

RabbitMq之死信队列_第1张图片

 


 

可以看出,两个队列都被标记了 "D 飞这个是 durable 的缩写,即设置了队列持久化。 queue.normal 这个队列还配置了 TTL、 DLX 和 DLK,其中 DLX 指的是x-dead-letter-routing-key 这个属性。

对于 RabbitMQ 来说, DLX 是一个非常有用的特性 。 它可以处理异常情况下,消息不能够被消费者正确消费(消费者调用了Basic.Nack 或者 Basic.Reject) 而被置入死信队列中的情况,后续分析程序可以通过消费这个死信队列中的内容来分析当时所遇到的异常情况,进而可以改善和优化系统。
 

 

参考资料:RabbitMq实战指南

你可能感兴趣的:(消息中间件)