本篇内容介绍了“RabbitMQ是怎么确定消息是否投递到队列中的”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!
公司主营业务:网站设计、成都网站建设、移动网站开发等业务。帮助企业客户真正实现互联网宣传,提高企业的竞争能力。创新互联建站是一支青春激扬、勤奋敬业、活力青春激扬、勤奋敬业、活力澎湃、和谐高效的团队。公司秉承以“开放、自由、严谨、自律”为核心的企业文化,感谢他们对我们的高要求,感谢他们从不同领域给我们带来的挑战,让我们激情的团队有机会用头脑与智慧不断的给客户带来惊喜。创新互联建站推出丘北免费做网站回馈大家。
1. 前言
在使用RabbitMQ消息中间件时,因为消息的投递是异步的,默认情况下,RabbitMQ会删除那些无法路由的消息。为了能够检出消息是否顺利投递到队列,我们需要相应的处理机制。今天就来验证一下相关的验证机制。
2. 消息投递失败
那么哪些情况消息会投递失败呢?RabbitMQ消息会先到达指定的交换机,然后由交换机路由到对应的队列。所以以下几种情况会导致消息投递失败。
投递的交换机不可用。
投递的交换机可用,但是没有匹配到队列。
3. 投递失败的处理机制
对应上面的两种情况,RabbitMQ提供了对应的解决方案。
ConfirmCallback
RabbitMQ提供了ConfirmCallback接口用于实现消息发送到RabbitMQ交换器后进行确认回调。
在Spring Boot中需要开启:
spring: rabbitmq: # 通常选择 correlated publisher-confirm-type:
通常有三种选择:
NONE ,禁用发布确认模式,是默认值。
CORRELATED,发布消息时会携带一个CorrelationData,被ack/nack时CorrelationData会被返回进行对照处理,CorrelationData可以包含比较丰富的元信息进行回调逻辑的处理。
SIMPLE,当被ack/nack后会等待所有消息被发布,如果超时会触发异常,甚至关闭连接通道。
这里我使用CORRELATED模式,声明一个ConfirmCallback并设置到RabbitTemplate中
rabbitTemplate.setConfirmCallback((correlationData, ack, cause) -> { // correlationData 可能为空 if (ack) { log.debug("消息发送到exchange成功,id: {}", correlationData.getId()); } else { log.debug("消息发送到exchange失败,原因: {}", cause); } });
当消息投递到一个不存在的交换机Exchange且ack=false时会输出日志:
- Publishing message [(Body:'"hello"' MessageProperties [headers={spring_listener_return_correlation=a088eb3f-a234-4e15-bb7a-3aa9a6f043e6, spring_returned_message_correlation=29975bc1-f363-4e3a-85ca-010d13888720, __TypeId__=java.lang.String}, contentType=application/json, contentEncoding=UTF-8, contentLength=7, deliveryMode=PERSISTENT, priority=0, deliveryTag=0])] on exchange [DIRECT_EXCHANGE1], routingKey = [DIRECT_ROUTING_KEY2] - 消息发送到exchange失败,原因: channel error; protocol method: #method(reply-code=404, reply-text=NOT_FOUND - no exchange 'DIRECT_EXCHANGE1' in vhost 'my_vhost', class-id=60, method-id=40)
这里实现的比较简单你可以增加一些消息投递到交换机失败后的操作处理逻辑。
ReturnCallback
ReturnCallback接口用于实现消息已经成功发送到RabbitMQ交换机,但没有匹配到队列时的回调。
在Spring Boot中需要同时开启:
spring: rabbitmq: publisher-returns: true template: mandatory: true
RabbitTemplate中的mandatory设置值优先级要高一些。
我们声明一个ReturnCallback并设置到RabbitTemplate中
rabbitTemplate.setMandatory(true); rabbitTemplate.setReturnCallback((message, replyCode, replyText, exchange, routingKey) -> { String correlationId = message.getMessageProperties() .getHeader(PublisherCallbackChannel.RETURNED_MESSAGE_CORRELATION_KEY); log.debug("消息:{} 发送失败, 应答码:{} 原因:{} 交换机: {} 路由键: {}", correlationId, replyCode, replyText, exchange, routingKey); });
当消息成功投递到交换机但是无法匹配到队列时:
- Publishing message [(Body:'"hello"' MessageProperties [headers={spring_listener_return_correlation=155648bd-fc3e-4c8b-a650-7b1ce720c7a6, spring_returned_message_correlation=7029ee49-357a-42fc-8532-dc41b4bb8e87, __TypeId__=java.lang.String}, contentType=application/json, contentEncoding=UTF-8, contentLength=7, deliveryMode=PERSISTENT, priority=0, deliveryTag=0])] on exchange [DIRECT_EXCHANGE], routingKey = [DIRECT_ROUTING_KEY2] - 消息:7029ee49-357a-42fc-8532-dc41b4bb8e87 发送失败, 应答码:312 原因:NO_ROUTE 交换机: DIRECT_EXCHANGE 路由键: DIRECT_ROUTING_KEY2 - 消息发送到exchange成功,id: 7029ee49-357a-42fc-8532-dc41b4bb8e87
从上面我们也可以看出ReturnCallback只处理投递到队列失败的情况,并不像ConfirmCallback既能处理失败的情况也能处理成功的情况。
“RabbitMQ是怎么确定消息是否投递到队列中的”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注创新互联网站,小编将为大家输出更多高质量的实用文章!
网站题目:RabbitMQ是怎么确定消息是否投递到队列中的
本文地址:http://scpingwu.com/article/jsccgh.html