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

Java中的RocketMQ消費(fèi)冪等詳解

 更新時(shí)間:2023年10月11日 10:36:04   作者:程序員勸退師-TAO  
這篇文章主要介紹了Java中的RocketMQ消費(fèi)冪等詳解,當(dāng)出現(xiàn)消費(fèi)者對某條消息重復(fù)消費(fèi)的情況時(shí),重復(fù)消費(fèi)的結(jié)果與消費(fèi)一次的結(jié)果是相同的,并且多次消費(fèi)并未對業(yè)務(wù)系統(tǒng)業(yè)務(wù)產(chǎn)生任何負(fù)面影響,那么這個(gè)消費(fèi)過程就是消息冪等,需要的朋友可以參考下

什么是消息冪等

當(dāng)出現(xiàn)消費(fèi)者對某條消息重復(fù)消費(fèi)的情況時(shí),重復(fù)消費(fèi)的結(jié)果與消費(fèi)一次的結(jié)果是相同的,并且多次消費(fèi)并未對業(yè)務(wù)系統(tǒng)業(yè)務(wù)產(chǎn)生任何負(fù)面影響,那么這個(gè)消費(fèi)過程就是消息冪等,在互聯(lián)網(wǎng)應(yīng)用中尤其在網(wǎng)絡(luò)不穩(wěn)定的情況下,消息很有可能會(huì)出現(xiàn)重復(fù)發(fā)送或者重復(fù)消費(fèi),如果重復(fù)的消息可能會(huì)影響業(yè)務(wù)處理,那么就應(yīng)該應(yīng)用消息冪等處理

消費(fèi)冪等的必要性

在互聯(lián)網(wǎng)應(yīng)用中,尤其在網(wǎng)絡(luò)不穩(wěn)定的情況下,消息隊(duì)列 RocketMQ 的消息有可能會(huì)出現(xiàn)重復(fù),這個(gè)重復(fù)簡單可以概括為以下情況:

  • 發(fā)送時(shí)消息重復(fù)

當(dāng)一條消息已被成功發(fā)送到服務(wù)端并完成持久化,此時(shí)出現(xiàn)了網(wǎng)絡(luò)閃斷或者客戶端宕機(jī),導(dǎo)致服務(wù)端對客戶端應(yīng)答失敗。 如果此時(shí)生產(chǎn)者意識到消息發(fā)送失敗并嘗試再次發(fā)送消息,消費(fèi)者后續(xù)會(huì)收到兩條內(nèi)容相同并且 Message ID 也相同的消息。

  • 投遞時(shí)消息重復(fù)

消息消費(fèi)的場景下,消息已投遞到消費(fèi)者并完成業(yè)務(wù)處理,當(dāng)客戶端給服務(wù)端反饋應(yīng)答的時(shí)候網(wǎng)絡(luò)閃斷。 為了保證消息至少被消費(fèi)一次,消息隊(duì)列 RocketMQ 的服務(wù)端將在網(wǎng)絡(luò)恢復(fù)后再次嘗試投遞之前已被處理過的消息,消費(fèi)者后續(xù)會(huì)收到兩條內(nèi)容相同并且 Message ID 也相同的消息。

  • Rebalance時(shí)消息重復(fù)(包括但不限于網(wǎng)絡(luò)抖動(dòng)、Broker 重啟以及訂閱方應(yīng)用重啟)

當(dāng)消息隊(duì)列 RocketMQ 的 Broker 或客戶端重啟、擴(kuò)容或縮容時(shí),會(huì)觸發(fā) Rebalance,此時(shí)消費(fèi)者可能會(huì)收到重復(fù)消息。

通用解決方案

兩要素

冪等性方案的設(shè)計(jì)中涉及兩項(xiàng)要輸,冪等令牌,與唯一性處理,只要充分利用好者兩要素,就可以設(shè)計(jì)出好的冪等解決方案。

  • 冪等令牌:是生產(chǎn)者和消費(fèi)者兩者中的既定令牌,通常具有唯一業(yè)務(wù)標(biāo)識的字符串
  • 唯一性處理:服務(wù)端通過采用一定的算法策略,保證同一個(gè)業(yè)務(wù)邏輯下不會(huì)被重復(fù)執(zhí)行成功多次

解決方案

對于常見的系統(tǒng),冪等性操作的通用性解決方案如下:、

  1. 首先通過緩存去重,在緩存中如果已經(jīng)存在某冪等令牌,則說明本次操作是重復(fù)性操作,若緩存中沒有命中,則進(jìn)入下一步
  2. 在唯一性處理之前,先在數(shù)據(jù)庫中查詢冪等令牌作為索引的數(shù)據(jù)是否存在,若存在,則說明本次操作為重復(fù)性操作,若不存在進(jìn)行下一步。
  3. 在同一事務(wù)中完成三項(xiàng)操作:唯一性處理后,將冪等令牌寫到緩存中,并將冪等令牌作為唯一索引的數(shù)據(jù)寫入DB中

設(shè)置業(yè)務(wù)key

因?yàn)?Message ID 有可能出現(xiàn)沖突(重復(fù))的情況,所以真正安全的冪等處理,不建議以 Message ID 作為處理依據(jù)。 最好的方式是以業(yè)務(wù)唯一標(biāo)識作為冪等處理的關(guān)鍵依據(jù),而業(yè)務(wù)的唯一標(biāo)識可以通過消息 Key 進(jìn)行設(shè)置:

Message message = new Message();
message.setKey("ORDERID_100");
SendResult sendResult = producer.send(message);

訂閱方收到消息時(shí)可以根據(jù)消息的 Key 進(jìn)行冪等處理:

consumer.subscribe("ons_test", "*", new MessageListener() {
    public Action consume(Message message, ConsumeContext context) {
        String key = message.getKey()
        // 根據(jù)業(yè)務(wù)唯一標(biāo)識的 key 做冪等處理
    }
});

注意:

消息重復(fù)消費(fèi)的情況是不能去避免的,我們需要考慮的就是在有重復(fù)消息的情況之下,怎么取保證冪等性,那么在保證冪等性的有一個(gè)關(guān)鍵,就是在發(fā)送消息的時(shí)候攜帶一個(gè)業(yè)務(wù)Key,然后在接收到消息后先去獲得這個(gè)業(yè)務(wù)Key,然后在消費(fèi)方的數(shù)據(jù)庫當(dāng)中判斷一下這個(gè)業(yè)務(wù)Key所對應(yīng)的消息有沒有消費(fèi)過,如果沒有消費(fèi)過就接著消費(fèi),消費(fèi)完了在數(shù)據(jù)中存儲一下,或者緩存數(shù)據(jù)庫中存儲也行,如果當(dāng)前業(yè)務(wù)Key對應(yīng)的消息已經(jīng)消費(fèi)過,那么直接舍棄即可!

支付實(shí)例場景

  • 當(dāng)支付請求到達(dá)后,首先在Redis中獲取key作為支付流水號的緩存value,若value不為空,則說明本次支付是重復(fù)操作,業(yè)務(wù)系統(tǒng)直接返回調(diào)用側(cè)重復(fù)支付標(biāo)識,若value為空,則進(jìn)入下一步操作
  • DBMS中根據(jù)支付流水號查詢是否存在相應(yīng)的實(shí)例,若存在,則說明本次支付是重復(fù)操作,業(yè)務(wù)系統(tǒng)直接返回調(diào)用側(cè)支付標(biāo)識,若不存在,則說明本次操作是首次操作,進(jìn)入下一步完成唯一性處理
  • 在分布式系統(tǒng)中完成三項(xiàng)操作
    • 完成支付任務(wù)
    • 將當(dāng)前支付流水號作為key,任意字段作為value,寫入到Redis緩存中
    • 將當(dāng)前支付流水號作為主鍵,與其他相關(guān)數(shù)據(jù)共同寫入到DBMS中

到此這篇關(guān)于Java中的RocketMQ消費(fèi)冪等詳解的文章就介紹到這了,更多相關(guān)RocketMQ消費(fèi)冪等內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 如何使用會(huì)話Cookie和Java實(shí)現(xiàn)JWT身份驗(yàn)證

    如何使用會(huì)話Cookie和Java實(shí)現(xiàn)JWT身份驗(yàn)證

    這篇文章主要介紹了如何使用會(huì)話Cookie和Java實(shí)現(xiàn)JWT身份驗(yàn)證,幫助大家更好的理解和使用Java,感興趣的朋友可以了解下
    2021-03-03
  • Java,JSP,Servlet獲取當(dāng)前工程路徑(絕對路徑)問題解析

    Java,JSP,Servlet獲取當(dāng)前工程路徑(絕對路徑)問題解析

    這篇文章主要介紹了Java,JSP,Servlet獲取當(dāng)前工程路徑(絕對路徑)問題解析,需要的朋友可以參考下。
    2017-09-09
  • 淺談Java中注解Annotation的定義、使用、解析

    淺談Java中注解Annotation的定義、使用、解析

    下面小編就為大家?guī)硪黄獪\談Java中注解Annotation的定義、使用、解析。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-04-04
  • MyBatis使用resultMap如何解決列名和屬性名不一致

    MyBatis使用resultMap如何解決列名和屬性名不一致

    這篇文章主要介紹了MyBatis使用resultMap如何解決列名和屬性名不一致的問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-01-01
  • Mybatis velocity腳本的使用教程詳解(推薦)

    Mybatis velocity腳本的使用教程詳解(推薦)

    很多朋友不清楚在mybatis可以使用各種腳本語言來定義Mapper文件里面的動(dòng)態(tài)SQL;目前mybatis支持的腳本語言有XML(默認(rèn)的);Velocity和Freemarker三種。下面通過本文給大家介紹Mybatis velocity腳本的使用,一起看看吧
    2016-11-11
  • SpringBoot 調(diào)度任務(wù)及常用任務(wù)表達(dá)式

    SpringBoot 調(diào)度任務(wù)及常用任務(wù)表達(dá)式

    這篇文章主要介紹了SpringBoot 調(diào)度任務(wù)及常用任務(wù)表達(dá)式,需要的朋友可以參考下
    2017-12-12
  • SpringBoot程序的打包與運(yùn)行的實(shí)現(xiàn)

    SpringBoot程序的打包與運(yùn)行的實(shí)現(xiàn)

    本文主要介紹了SpringBoot程序的打包與運(yùn)行的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • Java注解簡單使用實(shí)例解析

    Java注解簡單使用實(shí)例解析

    這篇文章主要介紹了Java注解簡單使用實(shí)例解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-06-06
  • mybatis返回結(jié)果為Map問題

    mybatis返回結(jié)果為Map問題

    這篇文章主要介紹了mybatis返回結(jié)果為Map問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2025-04-04
  • spring調(diào)度注解@Scheduled方式(含分布式)

    spring調(diào)度注解@Scheduled方式(含分布式)

    文章介紹了Java中任務(wù)調(diào)度的幾種常見方法,包括JDK原生的Timer、ScheduledThreadPoolExecutor和Spring的@Scheduled注解,文章還討論了如何在分布式環(huán)境中實(shí)現(xiàn)任務(wù)調(diào)度,并介紹了一些開源的分布式任務(wù)調(diào)度解決方案,如Quartz和XXL-JOB
    2024-11-11

最新評論