消息中間件的可靠性消息傳遞,是消息中間件領域很是重要的方案落實問題(在這以前的MQ理論,MQ選型是抽象層次更高的問題,這裏不談)。html
而且這個問題與平常開發是存在較大的關聯的。能夠這麼說,凡是使用了MQ的,機會都要考慮這個問題。固然也有一些原始數據採集,日誌數據收集等應用場景對此沒有太高要求。可是大多數的業務場景,對此仍是有着較高要求的。好比訂單系統,支付系統,消息系統等,你弄丟一條消息,嘿嘿。數據庫
網上對於這方面的博客,大多從單一MQ,或者乾脆就是在論述MQ。我不喜歡這樣的論述,這樣的論述太過侷限,也過於拖沓。api
此次,主要從理論方面論證消息的可靠性傳遞的落實。具體技術,都是依據這些理論的,具體實現都差很少。不過爲了便於你們理解,我在文中會以RabbitMq,Kafka這兩個主流MQ稍做舉例。服務器
在平常開發中,我更傾向於在具體開發前,先整理思路,走通理論,再開始編碼。畢竟,若是連理論都走不一樣,還談什麼編碼。網絡
另外,我按照消息可靠性層次逐步推動,造成相應的目錄,但願你們喜歡(由於我認爲,相較網上這方面現有博客的目錄,這樣的目錄更合理,更人性化)。併發
這裏簡單談一些有關消息可靠性傳遞的理論。框架
消息在消息系統(生產者+MQ+消費者),其消費的次數,無非一下三種狀況:異步
這也表明着消息系統的消息可靠性的三個層次:分佈式
實現上述三個層次,須要逐步從三個方面考慮:工具
上述三個層次,對系統的性能損耗,系統複雜度等都是逐步上升的。
固然,咱們首先,須要瞭解這三個層次分別如何實現。
再在實際開發中,根據須要,靈活選取合適方案。
這個方案是最簡單的,只要確保消息系統的正確運做,以及系統的連通性便可。在正常狀況下,能夠保證絕大部分數據的可靠性傳遞。可是仍舊存在極小數據的丟失,而且數據的丟失會由於消息隊列的選擇,以及消息併發量,而受到影響。
能夠應用於日誌上傳這樣對消息可靠性要求低的應用場景。
若是數據量不大的狀況下,推薦使用RabbitMQ,其消息可靠性在地數據量下,是最可靠的。可是在達到萬級併發時,會存在消息丟失,丟失的比例能夠達到千分之一。
若是數據量較大的狀況下,要麼採用集羣。要麼就採用Kafk(Kafka可支持十萬級併發)
通常來講,這種消息可靠性多見於項目初建,或相似日誌採集,原始數據採集這樣的特定場景。
這個方案開始利用MQ提供的特定機制,來提升消息傳遞的可靠性。
該方案的實現組成,由如下三個方面構成:
經過以上三個方面的落實,確保可消息必定被下游服務消費。
消息的可靠生產,是經過回調確認機制,確保消息必定被消息服務器接收。
消息生產,發送給消息服務器後,消息服務器會返回一個確認信息,表示數據正常接收。
若是生產者在必定時間內沒有接收到確認信息,就會觸發重試機制,進行消息的重發。
如RabbitMq的comfirm機制,Kafka的acks機制等。
RabbitMq的confirm機制存在三個模式:
這三個模式,看名稱就能夠知道具體做用了。若是但願瞭解具體代碼落實,詳見RabbitMQ事務和Confirm發送方消息確認——深刻解讀,其中確認機制寫得較爲簡潔。
至於Kafka的acks機制,一樣存在三個模式:
說到這裏,簡單說一下,上述的操做可能形成消息的重複生產。
最簡單的例子,消息成功發送,可是對應的消息確認信息因爲網絡波動而丟失。那麼生產者就會重複發送該消息,因此消息服務器接收到了兩條相同消息,故產生了消息的重複生產。
另外,上述的重試,都是存在響應時長判斷(超出1min,就認爲數據丟失),以及重試次數限制(超過5次,就不進行重試。不然,大量重試數據可能會拖垮整個服務)。
消息的可靠存儲,是確保消息在消息服務器通過,或者說堆積時不會由於宕機,網絡等情況,丟失消息。
網上不少博客在論述消息的可靠性傳遞時,經常把這點遺漏。由於他們理所固然地認爲消息隊列已經經過集羣等實現了消息隊列服務的可用性,故消息的可靠性存儲也就實現了。
可是這裏存在兩個問題。第一,可靠性不等於可用性。第二,消息的可靠存儲,做爲消息可靠性傳遞的一部分,是不可缺失的。
可用性:確保服務的可用。即對應的服務,能夠提供服務。
可靠性:確保服務的正確。即對應的服務,提供的是正確的服務。
區別:我瀏覽淘寶,淘寶頁面打不開,這就涉及了可用性問題(可用性計算公式:可用時間/所有時長*100%)。而我瀏覽淘寶,查詢訂單,給我顯示的是別人的訂單,這就涉及了可靠性問題。
另外這裏再糾正一點,可靠性並不依賴於可用性。即便我打不開淘寶頁面,我也不能說淘寶提供訂單查詢就有問題(只是若是沒有了可用性,談論可靠性是很是沒有意義的。畢竟都用不了了,誰還關心其內容是否正確呢,都看不到)
消息隊列的可用性,是經過多個節點構成集羣,避免單點故障,從而提高可用性。
消息隊列的可靠存儲,是經過備份實現(這裏不糾結備份如何確保正確)的。如RabbitMq集羣的MemNode與DiskNode,又或者Kafka的replication機制等。
消息的可靠消費,就是確保消息被消費者獲取,並被成功消費。避免因爲消息丟失,或者消費者宕機而形成消息消費不成功,最終形成消息的丟失(由於RabbitMq服務器在認爲消息被成功消費後,將對應數據刪除或標記爲「已消費」)。
至於消息的可靠消費,核心理念仍是重試,重試,再重試。不過具體的實現就八仙過海,各顯神通了。
這裏分別說一下RabbitMq,Kafka,Rocket三者對於可靠消費的處理:
提供ack機制。默認是auto,直接在拿到消息時,直接ack。確保了消息到達了消費者,可是沒法解決消費者消費失敗這樣的問題。
實際開發中,爲了確保消息的可靠消費,通常會設置爲munal,只有在程序正確運行後,纔會調用對應api,表示消息正確消費。
因爲Kafka的消息是落地到硬盤文件的,並且Kafka的消息分發方式是pull的,因此消息的拉取是經過offset機制去確認對應位置消息的。
固然,Kafka的offset默認是自動提交的(可經過nable_auto_commit與auto_commit_interval_ms控制)。
因此消費者調用服務失敗等緣由,能夠經過手動offset提交,來實現對數據的重複消費(甚至是歷史數據的消費),也就能夠在消費失敗時對同一消息進行再消費。
若是是消費者宕機等緣由,因爲Kafka服務器沒有收到對應的offset提交,因此認爲那條消息沒有被消費成功,故返回的依舊是那條消息。
其實RocketMq的處理有些相似Kafka確認機制+RabbitMq死信隊列的感受。
首先,消費者從RocketMq拉取消息,若是成功消費,就返回確認消息。
若是未成功消費,就嘗試從新消費。
嘗試消費必定次數後(如5次),就會將該消息發送之RocketMq中的重試隊列。
若是遇到消費者宕機的狀況,RocketMq會認爲該消息未成功消費,會被其餘消費者繼續消費。
其實在RabbitMq的可靠性消費時,咱們也會將屢次消費失敗的數據保存下來,便於後期修復等。不過保存的方式由不少種,日誌,數據庫,消息隊列等。而RocketMq則給出了具體的落實方案。
上述的操做,可能形成消息的重複消費。
最簡單的例子,消息成功被消費者消費,可是消費者還沒來得及發送確認信息,就宕機了。
消息隊列因爲沒有收到確認消息,認爲該條消息還沒有被消息,就將該消息交由其餘消費者繼續消費。
這個方案,就是經過MQ之外的應用程序,來進行擴展,最終達到消息準確消費的目的。
那麼爲何不將這個功能,囊括在MQ中呢?
我的認爲有四個方面的考慮:
消息存儲部分的準確存儲,不應咱們來操心,因此只闡述消息生產與消息消費兩個部分。
綜上來看,就是消息發出後,到生產者消息確認信息的處理之間,出現各類意外,致使重複生產。
綜上來看,就是消息已經被消費後,到消息隊列服務器進行確認消息處理之間,出現各類意外,致使重複消費。
解決方案:messageId+冪等
準確來講,解決方案的核心是冪等,而messageId是做爲輔助手段的。
冪等,簡單說明一下,就是屢次操做與單次操做對系統狀態的影響是一致的。
如
i = 1;
就是冪等操做,由於不管進行幾回,i的值都沒有變化。
而
i++;
則不是冪等操做,由於i的值與執行次數息息相關。
故經過冪等操做來確保同一條消息,不被執行屢次。
可是,消費者如何肯定是否爲同一條消息呢?
有的消息體存在惟一性字段,如orderId等。但有的消息並無這樣的惟一性字段。
因此須要一個專門的字段,來表示惟一性,而且與業務消息解耦。這就是messageId。
既能夠採用消息體的惟一性字段(能夠是單一字段,也能夠是組合字段),也能夠經過特定方式生成對應標識(分佈式系統,須要注意不一樣實例生產者產生相同標識的可能,詳見分佈式全局惟一ID的實現)。
具體的生成狀況,就不在這裏贅述了。
先來一張大圖(這種事情,圖片展現最直觀了),展現一下流程:
(圖片是絕對清晰的。看不清圖片的朋友,請將圖片在新頁面打開,或下載。說實話,來到新公司,首先提高的就是畫圖能力。囧)
簡單說一下流程,你們能夠對照着上圖,看一下:
上述中提到的補償機制,實際上是相似事務中的一個操做。經過一個定時任務,定時巡檢數據庫處於sending狀態的message,並經過生產者極性發送(因此message通常都保存source,target等信息)。
之因此會有sending狀態的message,就是由於存在生產者消息發送出去了,還沒收到生產確認信息,結果生產者實例本身宕機的狀況。
至於補償機制的定時任務,是一個很是簡單的實現,這裏就再也不贅述了。
這裏進行的操做是針對非冪等的操做。
若是是冪等操做,則能夠直接進行。畢竟屢次執行與單次執行對數據庫的影響是一致的。
可是注意冪等操做在部分場景下無效的問題(時間影響上),如「餘額 = 1k」的操做對於數據庫而言是冪等的,可是在兩次「餘額 = 1k」操做間,有一個「餘額 = 2k」的操做,則會發生問題(丟失了「餘額 = 2k」操做)。固然,這種相似ABA問題,徹底能夠引入版本號,來進行解決。
綜上,仍是推薦採用如下解決方法,流程較爲簡單:
至此,消息的準確傳遞就完成了。
消息可靠性傳遞的發展過程,也體現了人們對消息中間件功能的一步步追求,更是體現了工程師們解決問題的思路。
不少時候,咱們會遇到不少問題,甚至使人感到雜亂不堪,無從下手。這個時候,最好的辦法就是靜下心來,對它們進行劃分(按照重要程度,緊迫度,實現難度),再進行一個長期規劃,一步步來解決。每每這個時候,動動筆,在筆記上列下清單,會是一個不錯的辦法。
其中消息的準確傳遞,涉及一些事務相關的內容。也許有人已經聯想到,消息隊列是否能夠做爲分佈式事務的一種手段呢?我會在以後的博客中,來闡述分佈式事務這一重要主題。
若是有什麼問題或想法,能夠私信或@我。
願與諸君共進步。