JAVA面試如何保證消息不被重復消費?
今天,讓我們一起看看保證 MQ 的冪等性有哪些方案。
01. 冪等性的概念先說說什么是冪等性。
當用戶對同一操作請求了一次或者多次(消息發送或接收了多次),最終的結果是一致的,并不會因為多次請求產生副作用;比如同一個訂單支付了兩次,最后應該只扣客戶一次錢。
查詢和刪除:查詢具有天然的冪等性,在數據不變的前提下,相同查詢條件查詢一次和查詢多次的結果都是一樣的;刪除也一樣,相同的條件刪除一次和刪除多次,可能刪除的數據量不一樣,但是數據庫中的數據不會因為執行了多次刪除而不同。
新增和修改:如果不做冪等性處理,可能就會產生問題;執行多次新增操作,可能會導致一模一樣的數據產生了多條(主鍵自動生成);修改操作,如果只是把某些字段更新成固定的值,不會有冪等性問題,但是如果新值要在舊值上做處理做計算,如增加多少、減少多少,那么多次執行的結果就會有差異。
02. MQ 消息出現非冪等的情況1. 生產者發送消息給 MQ,為了保證消息可達,通常會使用超時重傳機制,但是如果生產者的消息已經發出去,但是由于網絡原因未收到確認信息,那么可能會進行重發,最終造成了消息的重復發送。
2. 消費者消費的過程也類似,消費者接受到 MQ 的消息,但是 MQ 未收到確認信息,那么該條消息可能會重新發送給其他消費者,或者網絡恢復再次發送給消費者,最終造成重復消費。
03. 解決方法1. 唯一索引
使用唯一索引,可以有效的防止新增臟數據:當表中存在唯一索引的時候,并發新增相同數據的時候就會報錯,不過這在單庫單表的時候才有效,如果項目數據量很大,采用了分庫分表的策略,就不能再通過數據庫的唯一性索引來解決冪等性的問題了。
2. 全局唯一 ID
每條消息,都攜帶一個全局的唯一 ID,消費者接收到消息,在執行前判斷這個 ID 是否已經在本地存在,如果不存在,則執行交易后記錄 ID(存到數據庫或Redis中,表示該交易已經執行);如果已經存在,表示消息已經消費掉了,不能再次執行。
許多分布式架構中,生成全局唯一 ID 都會被作為一個基礎的微服務,當然這個服務的可靠性要求極高,或者可以使用全局唯一 ID 算法,由每個應用自己生成。不過總的來說,引入全局唯一 ID 這個方案,實現起來還是非常繁瑣的。
3. 業務狀態
保證消息的冪等性,也可以結合業務流程考慮,比如很多業務流程每一步都是有狀態的,比如網上購物可能會有:訂單創建、付款、發貨,那么付款之前保單狀態為“待付款”,付款之后可以將保單的狀態修改為“待發貨”;那么如果發起重復扣款的話,第二次扣款的時候保單狀態已經變化了,就會扣款失敗。
4. 去重表
如果業務中有唯一性的標識時,可以使用去重表,把這個唯一性的表示保存到去重表中,如果重復插入,那么會被校驗住。
比如上面的場景,一個訂單只會付款一次,那么在付款的時候,把訂單號作為唯一性的標識,保存到去重表中,可以保證付款操作只會發生一次;這個方法也用到了唯一 ID,不過和全局唯一 ID 不同,這個唯一 ID 是針對具體業務的。
我將持續分享Java開發、架構設計、程序員職業發展等方面的見解,希望能得到你的關注。