欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

RabbitMQ消息確認(rèn)機(jī)制剖析

 更新時(shí)間:2022年08月18日 10:22:59   作者:劍圣無(wú)痕  
這篇文章主要為大家介紹了RabbitMQ消息確認(rèn)機(jī)制剖析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

前言

上一章講解了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)

    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搭建基本骨架

    這篇文章主要介紹了詳解mall整合SpringBoot+MyBatis搭建基本骨架,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-08-08
  • Java編程一維數(shù)組轉(zhuǎn)換成二維數(shù)組實(shí)例代碼

    Java編程一維數(shù)組轉(zhuǎn)換成二維數(shù)組實(shí)例代碼

    這篇文章主要介紹了Java編程一維數(shù)組轉(zhuǎn)換成二維數(shù)組,分享了相關(guān)代碼示例,小編覺得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下
    2018-01-01
  • Spring @Import注解的使用

    Spring @Import注解的使用

    @Import注解算是SpringBoot自動(dòng)配置原理中一個(gè)很重要的注解,本文介紹了該注解的源碼分析及使用方法,感興趣的朋友可以了解下
    2021-05-05
  • datatables 帶查詢條件java服務(wù)端分頁(yè)處理實(shí)例

    datatables 帶查詢條件java服務(wù)端分頁(yè)處理實(shí)例

    本篇文章主要介紹了datatables 帶查詢條件java服務(wù)端分頁(yè)處理實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • 基于RabbitMQ幾種Exchange 模式詳解

    基于RabbitMQ幾種Exchange 模式詳解

    下面小編就為大家?guī)?lái)一篇基于RabbitMQ幾種Exchange 模式詳解。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-11-11
  • 使用Java實(shí)現(xiàn)一個(gè)能保留計(jì)算過(guò)程的計(jì)算器

    使用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
  • java向文件末尾添加內(nèi)容示例分享

    java向文件末尾添加內(nèi)容示例分享

    本文為大家提供一個(gè)java向文件末尾添加內(nèi)容的示例分享,大家參考使用吧
    2014-01-01
  • 對(duì)dbunit進(jìn)行mybatis DAO層Excel單元測(cè)試(必看篇)

    對(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
  • 快速搭建springboot項(xiàng)目(新手入門)

    快速搭建springboot項(xiàng)目(新手入門)

    本文主要介紹了快速搭建springboot項(xiàng)目(新手入門),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-07-07

最新評(píng)論