Return Listener用于处理一些不可路由的消息!
我们的消息生产者,通过指定一个ExChange和Routingkey,把消息送达到某一个队列种去,然后我们的消费者监听队列,进行消费处理操作!
但是在某些情况下,如果我们在发送消息的时候,当前的Exchange不存在或者指定的路由key路由不到,这个时候如果我们需要监听种不可达的消息,就要使用Return Listener!
在基础的API种有一个关键的配置项:Mandatory:如果为true,则监听器会接受到路由不可达的消息,然后进行后续处理,如果为false,那么broker端自动删除该消息!
消费者代码
package com.star.movie.returnlistener;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.QueueingConsumer;
import com.star.movie.common.Constant;
import com.rabbitmq.client.QueueingConsumer.Delivery;
/**
* @Description:消费者
* @author:kaili
* @Date: 2019-04-22 17:41
**/
public class ReturnConsumer {
public static void main(String[] args) throws Exception{
Connection connection = Constant.getConnection();
Channel channel = connection.createChannel();
String exchangeName = "test_return_exchange";
String routingKey = "return.#";
String queueName = "test_return_queue";
channel.exchangeDeclare(exchangeName, "topic", true, false, null);
channel.queueDeclare(queueName, true, false, false, null);
channel.queueBind(queueName, exchangeName, routingKey);
QueueingConsumer queueingConsumer = new QueueingConsumer(channel);
channel.basicConsume(queueName, true, queueingConsumer);
while(true){
Delivery delivery = queueingConsumer.nextDelivery();
String msg = new String(delivery.getBody());
System.err.println("消费者: " + msg);
}
}
}
生产者代码快
package com.star.movie.returnlistener;
import com.rabbitmq.client.AMQP;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ReturnListener;
import com.star.movie.common.Constant;
import java.io.IOException;
/**
* @Description:生产者
* @author:kaili
* @Date: 2019-04-22 17:41
**/
public class ReturnProducer {
public static void main(String[] args) throws Exception{
Connection connection = Constant.getConnection();
Channel channel = connection.createChannel();
String exchange = "test_return_exchange";
String routingKey = "return.save";
String routingKeyError = "abc.save";
String msg = "Hello RabbitMQ Return Message";
channel.addReturnListener(new ReturnListener() {
@Override
public void handleReturn(int replyCode, String replyText, String exchange, String routingKey,
AMQP.BasicProperties properties, byte[] body) throws IOException {
System.err.println("---------handle return----------");
System.err.println("replyCode: " + replyCode);
System.err.println("replyText: " + replyText);
System.err.println("exchange: " + exchange);
System.err.println("routingKey: " + routingKey);
System.err.println("properties: " + properties);
System.err.println("body: " + new String(body));
}
});
//发送送达不到的消息mandatory属性为true
// channel.basicPublish(exchange, routingKeyError, false, null, msg.getBytes());
//发送送达不到的消息mandatory属性为false
channel.basicPublish(exchange, routingKeyError, true, null, msg.getBytes());
//发送正常消息
// channel.basicPublish(exchange, routingKey, true, null, msg.getBytes());
}
}
step 1 启动消费者代码
生成test_return_exchange交换机,绑定test_return_queue队列,绑定健为return.#,类型为topic
step 2 启动消费者代码
使用发送送达不到的消息mandatory属性为false这种发送方式,观察控制台打印消息。
消息发送失败,没有路由到对应的队列,在return监听代码中打印了相关消息路由的消息。
step 3 修改消费者代码再次启动
使用发送送达不到的消息mandatory属性为true这种发送方式,观察控制台打印消息。
生产者控制台没有任何消息
观察消费者控制台
消费者控制台也没有任何消息,说明该消息未发送到Broker,return监听也未接收到任何消息。这是因为Mandatory属性设置开启和关闭不同下的不同处理结果。