第四章----SpringBoot+RabbitMQ發送確認和消費手動確認機制

1. 配置RabbitMQ

# 發送確認
spring.rabbitmq.publisher-confirms=true
# 發送回調
spring.rabbitmq.publisher-returns=true
# 消費手動確認
spring.rabbitmq.listener.simple.acknowledge-mode=manual

2. 生產者發送消息確認機制

  • 其實這個也不能叫確認機制,只是起到一個監聽的作用,監聽生產者是否發送消息到exchange和queue。
  • 生產者和消費者代碼不改變。
  • 新建配置類 MQProducerAckConfig.java 實現ConfirmCallback和ReturnCallback接口,@Component注冊成組件。
  • ConfirmCallback只確認消息是否到達exchange,已實現方法confirm中ack屬性為標準,true到達,反之進入黑洞。
  • ReturnCallback消息沒有正確到達隊列時觸發回調,如果正確到達隊列不執行。
package com.fzb.rabbitmq.config;

import org.apache.commons.lang3.SerializationUtils;
import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.connection.CorrelationData;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;

/**
 * @Description 消息發送確認
 * <p>
 * ConfirmCallback  只確認消息是否正確到達 Exchange 中
 * ReturnCallback   消息沒有正確到達隊列時觸發回調,如果正確到達隊列不執行
 * <p>
 * 1. 如果消息沒有到exchange,則confirm回調,ack=false
 * 2. 如果消息到達exchange,則confirm回調,ack=true
 * 3. exchange到queue成功,則不回調return
 * 4. exchange到queue失敗,則回調return
 * @Author jxb
 * @Date 2019-04-04 16:57:04
 */
@Component
public class MQProducerAckConfig implements RabbitTemplate.ConfirmCallback, RabbitTemplate.ReturnCallback {

    @Autowired
    private RabbitTemplate rabbitTemplate;

    @PostConstruct
    public void init() {
        rabbitTemplate.setConfirmCallback(this);            //指定 ConfirmCallback
        rabbitTemplate.setReturnCallback(this);             //指定 ReturnCallback

    }

    @Override
    public void confirm(CorrelationData correlationData, boolean ack, String cause) {
        if (ack) {
            System.out.println("消息發送成功" + correlationData);
        } else {
            System.out.println("消息發送失敗:" + cause);
        }
    }

    @Override
    public void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey) {
        // 反序列化對象輸出
        System.out.println("消息主體: " + SerializationUtils.deserialize(message.getBody()));
        System.out.println("應答碼: " + replyCode);
        System.out.println("描述:" + replyText);
        System.out.println("消息使用的交換器 exchange : " + exchange);
        System.out.println("消息使用的路由鍵 routing : " + routingKey);
    }
}

3. 消費者消息手動確認

  • SpringBoot集成RabbitMQ確認機制分為三種:none、auto(默認)、manual

Auto:
1. 如果消息成功被消費(成功的意思是在消費的過程中沒有拋出異常),則自動確認
2. 當拋出 AmqpRejectAndDontRequeueException 異常的時候,則消息會被拒絕,且 requeue = false(不重新入隊列)
3. 當拋出 ImmediateAcknowledgeAmqpException 異常,則消費者會被確認
4. 其他的異常,則消息會被拒絕,且 requeue = true,此時會發生死循環,可以通過 setDefaultRequeueRejected(默認是true)去設置拋棄消息

  • 如設置成manual手動確認,一定要對消息做出應答,否則rabbit認為當前隊列沒有消費完成,將不再繼續向該隊列發送消息。

  • channel.basicAck(long,boolean); 確認收到消息,消息將被隊列移除,false只確認當前consumer一個消息收到,true確認所有consumer獲得的消息。

  • channel.basicNack(long,boolean,boolean); 確認否定消息,第一個boolean表示一個consumer還是所有,第二個boolean表示requeue是否重新回到隊列,true重新入隊。

  • channel.basicReject(long,boolean); 拒絕消息,requeue=false 表示不再重新入隊,如果配置了死信隊列則進入死信隊列。

  • 當消息回滾到消息隊列時,這條消息不會回到隊列尾部,而是仍是在隊列頭部,這時消費者會又接收到這條消息,如果想消息進入隊尾,須確認消息后再次發送消息。

channel.basicPublish(message.getMessageProperties().getReceivedExchange(),
                    message.getMessageProperties().getReceivedRoutingKey(), 
                    MessageProperties.PERSISTENT_TEXT_PLAIN,
                    message.getBody());
  • 延續上一章direct類型隊列為例,當消息出現異常,判斷是否回滾過消息,如否則消息從新入隊,反之拋棄消息。其中一個消費者模擬一個異常。
    @RabbitListener(bindings = {@QueueBinding(value = @Queue(value = "direct.queue"), exchange = @Exchange(value = "direct.exchange"), key = "HelloWorld")})
    public void getDirectMessage(User user, Channel channel, Message message) throws IOException {
        try {
            // 模擬執行任務
            Thread.sleep(1000);
            // 模擬異常
            String is = null;
            is.toString();
            // 確認收到消息,false只確認當前consumer一個消息收到,true確認所有consumer獲得的消息
            channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);
        } catch (Exception e) {
            if (message.getMessageProperties().getRedelivered()) {
                System.out.println("消息已重復處理失敗,拒絕再次接收" + user.getName());
                // 拒絕消息,requeue=false 表示不再重新入隊,如果配置了死信隊列則進入死信隊列
                channel.basicReject(message.getMessageProperties().getDeliveryTag(), false);
            } else {
                System.out.println("消息即將再次返回隊列處理" + user.getName());
                // requeue為是否重新回到隊列,true重新入隊
                channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, true);
            }
            //e.printStackTrace();
        }
    }

@RabbitListener(queues = "direct.queue")
    public void getDirectMessageCopy(User user, Channel channel, Message message) throws IOException {
        try {
            // 模擬執行任務
            Thread.sleep(1000);
            System.out.println("--jxb--MQConsumer--getDirectMessageCopy:" + user.toString());
            // 確認收到消息,false只確認當前consumer一個消息收到,true確認所有consumer獲得的消息
            channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);
        } catch (Exception e) {
            if (message.getMessageProperties().getRedelivered()) {
                System.out.println("消息已重復處理失敗,拒絕再次接收!");
                // 拒絕消息,requeue=false 表示不再重新入隊,如果配置了死信隊列則進入死信隊列
                channel.basicReject(message.getMessageProperties().getDeliveryTag(), false);
            } else {
                System.out.println("消息即將再次返回隊列處理!");
                // requeue為是否重新回到隊列,true重新入隊
                channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, true);
            }
            e.printStackTrace();
        }
    }

從執行結果來看,三條消息都調用了confirm方法,說明消息發送到了exchange,且沒有調用return方法,說明消息成功到達相應隊列。

getDirectMessageCopy方法成功消費掉“張三”這條消息,由于getDirectMessage方法模擬異常,所以第一次把“李四”從新入隊,此時getDirectMessageCopy繼續消費“王五”成功,getDirectMessage方法因李四已經從新入隊過,再次發生異常則拋棄消息。

輪詢分發

進一步挖掘你會發現,開始一共3條消息,有一條回滾消息總數變成了4條,每個消費者消費2條,所以兩個消費者是輪詢分配的。

  • 工作隊列有兩種工作方式:輪詢分發(默認)、公平分發即當某個消費者沒有消費完成之前不用再分發消息。
  • 修改配置文件
# 消費者每次從隊列獲取的消息數量。此屬性當不設置時為:輪詢分發,設置為1為:公平分發
spring.rabbitmq.listener.simple.prefetch=1

將第一個消費者模擬執行5秒,然后向數據庫增加一條數據,執行結果為:

公平分發

可以看到,getDirectMessageCopy執行了4次,getDirectMessage執行了1次,根據他們的消費能力來公平分發消息。


如果可以,我想回到10年前...

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,606評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,582評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,540評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,028評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,801評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,223評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,294評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,442評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,976評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,800評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,996評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,543評論 5 360
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,233評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,662評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,926評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,702評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,991評論 2 374