RabbitMQ消息確認(rèn)機(jī)制剖析
前言
上一章講解了RabbitMq的三種Exchange消息發(fā)送的模式,但是在默認(rèn)情況下RabbitMQ并不能保證消息是否發(fā)送成功,以及是否能被成功消費(fèi),為了保證消息在傳遞過(guò)程中不丟失,需要對(duì)消息進(jìn)行確認(rèn)機(jī)制,來(lái)提高消息的可靠性。
消息確認(rèn)
基本流程
說(shuō)明:
- 生產(chǎn)者發(fā)送消息到RabbitMQ Server后,RabbitMQ Server需要對(duì)生產(chǎn)者進(jìn)行消息Confirm確認(rèn)。
- 消費(fèi)者消費(fèi)消息后需要對(duì) RabbitMQ Server進(jìn)行消息ACK確認(rèn)。
消息確認(rèn)模式
RabbitMq提供了兩種消息發(fā)送者確認(rèn)模式分別為: ConfirmCallback確認(rèn)模式和 ReturnCallback退回模式。
ConfirmCallback確認(rèn)模式
@Component public class RabbitConfirmConfig implements ConfirmCallback { private Logger logger = LoggerFactory.getLogger(RabbitConfirmConfig.class); public void confirm(CorrelationData correlationData, boolean ack, String cause) { logger.info("數(shù)據(jù)內(nèi)容:{}",correlationData); logger.info("是否確認(rèn)成功:{}",ack); logger.info("錯(cuò)誤原因:{}",cause); if (!ack) { logger.info("exchange produce confirm message send error" + cause); } else { logger.info("exchange produce confirm message send success"); } } }
說(shuō)明:ConfirmCallback模式確認(rèn),需要重寫confirm接方法,此方法的三個(gè)參數(shù)分別為:CorrelationData、ack、cause
- CorrelationData:對(duì)象內(nèi)部只有一個(gè)id屬性,用來(lái)表示當(dāng)前消息的唯一性。
- ack:消息投遞狀態(tài),true表示投遞成功
- cause: 消息投遞失敗原因
雖然消息被broker接收到只能表示已經(jīng)到達(dá)MQ服務(wù)器,但是并不能保證消息一定會(huì)被投遞到目標(biāo) queue里。所以我們需要實(shí)現(xiàn)returnCallback來(lái)進(jìn)行相關(guān)處理。
ReturnCallback退回模式
@Component public class RabbitReturnConfig implements ReturnCallback { private Logger logger = LoggerFactory.getLogger(RabbitReturnConfig.class); public void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey) { logger.info("消息發(fā)送送到隊(duì)列信息:"); logger.info("發(fā)生消息:{}",message); logger.info("回應(yīng)碼:{}",replyCode); logger.info("回應(yīng)信息:{}",replyText); logger.info("交換機(jī):{}",exchange); logger.info("路由鍵:{}",routingKey); } }
說(shuō)明:實(shí)現(xiàn)接口ReturnCallback重寫returnedMessage()方法,方法有五個(gè)參數(shù)message(消息體)、replyCode(響應(yīng)code)、replyText(響應(yīng)內(nèi)容)、exchange(交換機(jī))、routingKey(路由鍵)。
消息發(fā)送者確認(rèn)
@Component public class MqConfirmProduce { @Autowired private RabbitTemplate rabbitTemplate; @Autowired private RabbitConfirmConfig rabbitConfirmConfig; @Autowired private RabbitReturnConfig rabbitReturnConfig; /** * * @param exchange 消息交互機(jī)名稱 * @param routeKey 消息路由鍵的名稱 * @param message 消息內(nèi)容 */ public void sendMessage(String exchange ,String routeKey,Object msg) { //確保消息發(fā)送失敗后可以重新返回到隊(duì)列中 rabbitTemplate.setMandatory(true); // 消費(fèi)者確認(rèn)收到消息后,手動(dòng)ack回執(zhí)回調(diào)處理 rabbitTemplate.setConfirmCallback(rabbitConfirmConfig); //消息投遞到隊(duì)列失敗回調(diào)處理 rabbitTemplate.setReturnCallback(rabbitReturnConfig); //保證消息唯一性 CorrelationData correlationData =new CorrelationData(UUID.randomUUID().toString()); //發(fā)送消息 rabbitTemplate.convertAndSend(exchange,routeKey,msg, message -> { message.getMessageProperties().setDeliveryMode(MessageDeliveryMode.PERSISTENT); return message; }, correlationData); } }
說(shuō)明:注意需要開啟消息確認(rèn)的配置:
rabbitmq: host: localhost port: 5672 username: guest password: guest virtual-host: / #開啟發(fā)送確認(rèn) publisher-confirms: true # 開啟發(fā)送失敗退回 publisher-returns: true listener: simple: # 手動(dòng)確認(rèn) acknowledge-mode: manual retry: enabled: true
消息接收者確認(rèn)
@Component @RabbitListener(queues = "testQueue") public class MqConfirmConsumer { private static final Logger logger = LoggerFactory.getLogger(MqConfirmConsumer.class); @RabbitHandler public void receive(String msg, Channel channel, Message message) throws IOException { logger.info("receive message content:{}",message); try { logger.info("開始消息確認(rèn)"); channel.basicAck(message.getMessageProperties().getDeliveryTag(), false); logger.info("消息確認(rèn)成功"); } catch (Exception e) { logger.error("消息確認(rèn)失敗,即將再次返回隊(duì)列中"); channel.basicNack(message.getMessageProperties().getDeliveryTag(), true, true); } } }
說(shuō)明:消息者確認(rèn)消息有三種模式,分別為basicAck、basicNack、basicReject。
basicAck模式
表示成功確認(rèn),使用此回執(zhí)方法后,消息會(huì)被rabbitmq broker刪除。
void basicAck(long deliveryTag, boolean multiple)
- deliveryTag:消息投遞序號(hào),
- multiple:是否批量確認(rèn),值為 true則會(huì)一次性ack所有小于當(dāng)前消息deliveryTag的消息。
basicNack模式
表示失敗確認(rèn),一般在消費(fèi)消息異常時(shí)用到此方法,可以將消息重新投遞入隊(duì)列。
void basicNack(long deliveryTag, boolean multiple, boolean requeue)
- deliveryTag:表示消息投遞序號(hào)。
- requeue: 表示消息是否重新入隊(duì)列,true表示重新投入隊(duì)列中。
- multiple:是否批量確認(rèn),true表示會(huì)一次性ack所有小于當(dāng)前消息deliveryTag的消息。
basicReject模式
basicReject:拒絕消息,與basicNack區(qū)別在于不能進(jìn)行批量操作,其他用法很相似。
void basicReject(long deliveryTag, boolean requeue)
- deliveryTag:消息投遞序號(hào)。
- requeue:值為true表示消息重新入隊(duì)列
測(cè)試
測(cè)試發(fā)送消息,消息發(fā)送者的確認(rèn)信息如下:
c.s.f.r.config.RabbitConfirmConfig - exchange produce confirm message send success
c.s.f.r.config.RabbitConfirmConfig - 數(shù)據(jù)內(nèi)容:CorrelationData [id=88ea47a5-726d-44c5-9839-1f2a6bf942ed]
c.s.f.r.config.RabbitConfirmConfig - 是否確認(rèn)成功:true
c.s.f.r.config.RabbitConfirmConfig - 錯(cuò)誤原因:null
c.s.f.r.config.RabbitConfirmConfig - exchange produce confirm message send success
消費(fèi)者的確認(rèn)信息如下:
receive message content:(Body:'this is test message' MessageProperties [headers={spring_listener_return_correlation=0fcefb6d-acea-4eb2-8484-e3a82f8c584f, spring_returned_message_correlation=88ea47a5-726d-44c5-9839-1f2a6bf942ed}, contentType=text/plain, contentEncoding=UTF-8, contentLength=0, receivedDeliveryMode=PERSISTENT, priority=0, redelivered=false, receivedExchange=testDirect, receivedRoutingKey=testDirectRouting, deliveryTag=2, consumerTag=amq.ctag-dOwkSPuI1e0HR_1Ufu3Erw, consumerQueue=testQueue])
c.s.f.r.consumer.MqConfirmConsumer - 開始消息確認(rèn)
c.s.f.r.consumer.MqConfirmConsumer - 消息確認(rèn)成功
消費(fèi)者確認(rèn)失敗
如果消息確認(rèn)在消費(fèi)者確認(rèn)失敗,那么消息將會(huì)重寫投遞導(dǎo)導(dǎo)消息隊(duì)列的首部。模擬消費(fèi)者確認(rèn)失敗場(chǎng)景:
@Component @RabbitListener(queues = "testQueue") public class MqConfirmConsumer { private static final Logger logger = LoggerFactory.getLogger(MqConfirmConsumer.class); @RabbitHandler public void receive(String msg, Channel channel, Message message) throws IOException { logger.info("receive message content:{}",message); try { logger.info("開始消息確認(rèn)"); int c=1/0; channel.basicAck(message.getMessageProperties().getDeliveryTag(), false); logger.info("消息確認(rèn)成功"); } catch (Exception e) { logger.error("消息確認(rèn)失敗,即將再次返回隊(duì)列中"); channel.basicNack(message.getMessageProperties().getDeliveryTag(), true, true); } } }
查看執(zhí)行結(jié)果:
c.s.f.r.consumer.MqConfirmConsumer - receive message content:(Body:'this is test message' MessageProperties [headers={spring_listener_return_correlation=0fcefb6d-acea-4eb2-8484-e3a82f8c584f, spring_returned_message_correlation=39d4cdd1-cbeb-4090-91ea-9e5d0bed785c}, contentType=text/plain, contentEncoding=UTF-8, contentLength=0, receivedDeliveryMode=PERSISTENT, priority=0, redelivered=false, receivedExchange=testDirect, receivedRoutingKey=testDirectRouting, deliveryTag=1, consumerTag=amq.ctag-e5GtG455pkm7eWfY3xGleg, consumerQueue=testQueue])
c.s.f.r.consumer.MqConfirmConsumer - 開始消息確認(rèn)
c.s.f.r.consumer.MqConfirmConsumer - 消息確認(rèn)失敗,即將再次返回隊(duì)列中
消息已經(jīng)重新返回隊(duì)列中。我們查看隊(duì)列信息具體如下:
說(shuō)明:我們可以看到消息為Unacked狀態(tài),消息又會(huì)重新會(huì)被消費(fèi),然后確認(rèn)失敗,又重新被消費(fèi),導(dǎo)致死循環(huán)。
解決辦法
針對(duì)這種情況,我們將如何處理呢?我們手動(dòng)確認(rèn)失敗后,并將消息持久入到MySQL中通過(guò)定時(shí)任務(wù)做補(bǔ)償。然后刪除消息隊(duì)列。具體修改如下:
@RabbitHandler public void receive(String msg, Channel channel, Message message) throws IOException { logger.info("receive message content:{}",message); try { logger.info("開始消息確認(rèn)"); int c=1/0; channel.basicAck(message.getMessageProperties().getDeliveryTag(), false); logger.info("消息確認(rèn)成功"); } catch (Exception e) { if (message.getMessageProperties().getRedelivered()) { logger.error("消息確認(rèn)失敗,拒絕處理"); //執(zhí)行持久化處理 channel.basicReject(message.getMessageProperties().getDeliveryTag(), false); } else { logger.error("消息確認(rèn)失敗,即將再次返回隊(duì)列中"); channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, true); } } }
修改后執(zhí)行結(jié)果如下:
總結(jié)
本文講解了RabbitMQ消息確認(rèn)機(jī)制,消息是否需要確認(rèn),我們需要根據(jù)業(yè)務(wù)的場(chǎng)景來(lái)分析,如有疑問(wèn),請(qǐng)隨時(shí)反饋,更多關(guān)于RabbitMQ消息確認(rèn)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
springboot整合RabbitMQ發(fā)送短信的實(shí)現(xiàn)
本文會(huì)和SpringBoot做整合,實(shí)現(xiàn)RabbitMQ發(fā)送短信,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05詳解mall整合SpringBoot+MyBatis搭建基本骨架
這篇文章主要介紹了詳解mall整合SpringBoot+MyBatis搭建基本骨架,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08Java編程一維數(shù)組轉(zhuǎn)換成二維數(shù)組實(shí)例代碼
這篇文章主要介紹了Java編程一維數(shù)組轉(zhuǎn)換成二維數(shù)組,分享了相關(guān)代碼示例,小編覺得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01datatables 帶查詢條件java服務(wù)端分頁(yè)處理實(shí)例
本篇文章主要介紹了datatables 帶查詢條件java服務(wù)端分頁(yè)處理實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06使用Java實(shí)現(xiàn)一個(gè)能保留計(jì)算過(guò)程的計(jì)算器
計(jì)算器是我們?nèi)粘I钪谐S玫墓ぞ咧?它能夠進(jìn)行基本的數(shù)學(xué)運(yùn)算,如加法、減法、乘法和除法,而在設(shè)計(jì)一個(gè)計(jì)算器時(shí),我們可以通過(guò)使用Java編程語(yǔ)言來(lái)實(shí)現(xiàn)一個(gè)簡(jiǎn)單的控制臺(tái)計(jì)算器,并且讓它能夠保留計(jì)算過(guò)程,文中有詳細(xì)的代碼示例,需要的朋友可以參考下2023-11-11對(duì)dbunit進(jìn)行mybatis DAO層Excel單元測(cè)試(必看篇)
下面小編就為大家?guī)?lái)一篇對(duì)dbunit進(jìn)行mybatis DAO層Excel單元測(cè)試(必看篇)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-05-05