三張表有重復字段_什么?搞不定Kafka重復消費?

5dc53e41d74f334b1b8e2fd9156adf9d.gif點戳藍字“架構之美”關注我們哦!

? 前言?

今天我們聊一個話題,這個話題大家可能在面試過程中,或者是工作當中經常遇到 ?如何保證 Kafka 消息不重復消費?我們在做開發的時候為了程序的健壯性,在使用 Kafka 的時候一般都會設置重試的次數,但是因為網絡的一些原因,設置了重試就有可能導致有些消息重復發送了(當然導致消息重復也有可能是其他原因),那么怎么解決消息重復這個問題呢?關于這個問題,我這兒提供了如下三種解決方案,供大家參考。

?解決方案

方案一? /? 保存并查詢

給每個消息都設置一個獨一無二的 key,消費的時候把 key 記錄下來,然后每次消費新的消息的時候都查詢一下,看當前消息的這個 key 是否消費過,如果沒有消費過才進行消費。(這種方式好想,但是其實實現起來一點也不簡單)ed439aa4e6e67e7f41c896288c2016e9.png方案二?/??利用冪等冪等(Idempotence)在數學上是這樣定義的,如果一個函數 f(x) 滿足:f(f(x)) = f(x),則函數 f(x) 滿足冪等性。這個概念被拓展到計算機領域,被用來描述一個操作、方法或者服務。一個冪等操作的特點是,其任意多次執行所產生的影響均與一次執行的影響相同。一個冪等的方法,使用同樣的參數,對它進行多次調用和一次調用,對系統產生的影響是一樣的。所以,對于冪等的方法,不用擔心重復執行會對系統造成任何改變。我們舉個例子?來說明一下。在不考慮并發的情況下,“將 X 老師的賬戶余額設置為 100 萬元”,執行一次后對系統的影響是,X 老師的賬戶余額變成了 100 萬元。只要提供的參數 100萬元不變,那即使再執行多少次,X 老師的賬戶余額始終都是 100萬元,不會變化,這個操作就是一個冪等的操作。再舉一個例子?,“將 X 老師的余額加 100 萬元”,這個操作它就不是冪等的,每執行一次,賬戶余額就會增加 100 萬元,執行多次和執行一次對系統的影響(也就是賬戶的余額)是不一樣的。所以,通過這兩個例子,我們可以想到如果系統消費消息的業務邏輯具備冪等性,那就不用擔心消息重復的問題了,因為同一條消息,消費一次和消費多次對系統的影響是完全一樣的。也就可以認為,消費多次等于消費一次。那么,如何實現冪等操作呢?最好的方式就是,從業務邏輯設計上入手,將消費的業務邏輯設計成具備冪等性的操作。但是,不是所有的業務都能設計成天然冪等的,這里就需要一些方法和技巧來實現冪等。下面我們介紹一種常用的方法:利用數據庫的唯一約束實現冪等。例如,我們剛剛提到的那個不具備冪等特性的轉賬的例子:將 X 老師的賬戶余額加 100 萬元。在這個例子中,我們可以通過改造業務邏輯,讓它具備冪等性。首先,我們可以限定,對于每個轉賬單每個賬戶只可以執行一次變更操作,在分布式系統中,這個限制實現的方法非常多,最簡單的是我們在數據庫中建一張轉賬流水表,這個表有三個字段:轉賬單 ID、賬戶 ID 和變更金額,然后給轉賬單 ID 和賬戶 ID 這兩個字段聯合起來創建一個唯一約束,這樣對于相同的轉賬單 ID 和賬戶 ID,表里至多只能存在一條記錄。這樣,我們消費消息的邏輯可以變為:“在轉賬流水表中增加一條轉賬記錄,然后再根據轉賬記錄,異步操作更新用戶余額即可。”在轉賬流水表增加一條轉賬記錄這個操作中,由于我們在這個表中預先定義了“賬戶 ID 轉賬單 ID”的唯一約束,對于同一個轉賬單同一個賬戶只能插入一條記錄,后續重復的插入操作都會失敗,這樣就實現了一個冪等的操作。5bdd6f4c355cad157c6362b7583130f5.png方案三 /??設置前置條件

為更新的數據設置前置條件另外一種實現冪等的思路是,給數據變更設置一個前置條件,如果滿足條件就更新數據,否則拒絕更新數據,在更新數據的時候,同時變更前置條件中需要判斷的數據。

這樣,重復執行這個操作時,由于第一次更新數據的時候已經變更了前置條件中需要判斷的數據,不滿足前置條件,則不會重復執行更新數據操作。

比如,剛剛我們說過,“將 X 老師的賬戶的余額增加 100 萬元”這個操作并不滿足冪等性,我們可以把這個操作加上一個前置條件,變為:“如果X老師的賬戶當前的余額為 500萬元,將余額加 100萬元”,這個操作就具備了冪等性。

對應到消息隊列中的使用時,可以在發消息時在消息體中帶上當前的余額,在消費的時候進行判斷數據庫中,當前余額是否與消息中的余額相等,只有相等才執行變更操作。

但是,如果我們要更新的數據不是數值,或者我們要做一個比較復雜的更新操作怎么辦?用什么作為前置判斷條件呢?更加通用的方法是,給你的數據增加一個版本號屬性,每次更數據前,比較當前數據的版本號是否和消息中的版本號一致,如果不一致就拒絕更新數據,更新數據的同時將版本號 +1,一樣可以實現冪等。

648e1c6ea18300c11e5e065e74e608cf.png

?最后

今天給大家提供的消息重復的解決方案,也參考了《消息隊列高手課》里的思路,大家如果有什么好的解決方案,歡迎討論!!大家加油!!e06571c62968f83fd025e6a9b96597fd.png

·end·

奈學教育科技

技術人成長之路的指路人

2b761542b0b724375d3be5fffac5a360.png

點贊支持技術創業?

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/530904.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/530904.shtml
英文地址,請注明出處:http://en.pswp.cn/news/530904.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

如何利用擴展歐幾里得算法求解不定方程_歐幾里德算法、拓展歐幾里德、中國剩余定理...

01.歐幾里德算法(Euclidean algorithm)(輾轉相除法)歐幾里德算法又稱輾轉相除法,主要是用于計算兩個整數a,b的最大公約數。簡單點說一下算法原理:兩個整數的最大公約數等于其中小的那個數跟大除以小余數的最…

mysql 先刪后增 更新_MySQL 高級操作——新增數據、更新數據、刪除數據、查詢數據...

新增數據多數據插入只要寫一次insert指令,但是可以插入多條記錄語法:insert into 表名 [(字段列表)] values (值列表1),(值列表2),(值列表3);主鍵沖突主鍵沖突,在有的表中,使用的是業務主鍵(字段有業務含義),但是往往在…

nltk和python的關系_NLTK學習筆記(一):語言處理和Python

目錄nltk資料下載import nltknltk.download()其中,download() 參數默認是all,可以在腳本里面加上nltk.download(需要的資料庫) 來進行下載文本和詞匯首先,通過from nltk.book import * 引入需要的內置9本書搜索文本上下文:Text.concordance(monstrous) &…

python七段數碼管倒計時_python實現七段數碼管和倒計時效果

8是典型的七段數碼管的例子,因為剛好七段都有經過,這里我寫的代碼是從1開始右轉。這是看Mooc視頻寫的一個關于用七段數碼管顯示當前時間# -*-coding:utf-8 -*-import turtle as timport timedef drawGap():t.penup()t.fd(5)def drawLine(draw):drawGap()…

rda分析怎么做_數量生態學筆記||冗余分析(RDA)

上一節數量生態學筆記||冗余分析(RDA)概述中,我們回顧了RDA的計算過程,不管這個過程我們有沒有理解透徹,我希望你能知道的是:RDA是響應變量矩陣與解釋變量之間多元多重線性回歸的擬合值矩陣的PCA分析。本節我們就是具體來看一個RD…

mysql 服務器管理員_mysql 查看數據庫管理員

mysql 查看數據庫管理員云服務器(Elastic Compute Service,簡稱ECS)是阿里云提供的性能卓越、穩定可靠、彈性擴展的IaaS(Infrastructure as a Service)級別云計算服務。云服務器ECS免去了您采購IT硬件的前期準備,讓您像使用水、電、天然氣等公共資源一樣…

python中有哪些重要的書寫規則_一文讀懂Python代碼的書寫規范

Python代碼的書寫規范1. 一致性的建議打破一條既定規則的兩個好理由當應用這個規則將導致代碼可讀性下降,即使對于某人來說他已經習慣于按照這條規則來閱讀代碼了為了和周圍的代碼保持一致而打破規則(也許是歷史原因)2. 代碼的布局縮進4個空格代碼行行最大長度 : 79字符推薦長度…

二進制文件mysql創表_MySQL_MYSQL中如何存取二進制文件,首先創建測試表testtable CREATE TA - phpStudy...

MYSQL中如何存取二進制文件首先創建測試表testtableCREATE TABLE testtable ( id INT(5) NOT NULL AUTO_INCREMENT PRIMARY KEY,filename CHAR(255),data LONGBLOB );將文件存入表中mysql_connect( "localhost", "root", "password"); //連接數據…

樹莓派 php mysql 中文_使用樹莓派(raspberry pi)搭建網站(nginx+php+mysql+ddclient)

標簽: 樹莓派 raspberrypi php 網站 mysql分類: Linux技術最近在研究學習PHP,有時候想隨時就學習,所以就決定搭建一個網站,隨時可以進行學習,因為要24小時在線,要低功耗和安靜,所以選…

mysql從庫應用負載_線上MySQL數據庫高負載的解決思路--再次論程序應用索引的重要性...

前言:過去的筆記整理而得,未免丟失,發布個人博客。[2012年的資料筆記]場景:數據庫的負載飆升,CPU高達99%。查看進程。通過猜測推理,定位了一些select語句363478427 | apps_read | 192.168.1.113:48945 …

python獲取方法的裝飾方法_python中的方法和裝飾器

[TOC]裝飾器python中的裝飾器(decorator)是在pep 318中被首次引入,它的本質是一個函數這個函數是接受其它參數為參數,并且用一個新的,修改后的函數作為替換,最常見的裝飾器就classmethod和staticmethoddef happy(f):return lambda…

一幫一python_[python]L1-030?一幫一?(15分)

L1-030 一幫一 (15分)“一幫一學習小組”是中小學中常見的學習組織方式,老師把學習成績靠前的學生跟學習成績靠后的學生排在一組。本題就請你編寫程序幫助老師自動完成這個分配工作,即在得到全班學生的排名后,在當前尚未分組的學生中&#xf…

java書面_Java程序猿的書面采訪String3

public class SameString {//思想二:每個字符都相應著自己的ASC碼,第一個思想的算法復雜度為O(nlogn)。一般能夠利用空間來減少時間復雜度//能夠開辟一個大小為256的數組空間,而且將256個數組元素都置為0,然后遍歷第一個字符串把字…

java fangfa_daicanfangfa java中的方法 剛入門的分不清帶參方法的作用和用處 這個可以詳細的講解如何使用帶參方法 - 下載 - 搜珍網...

第14章 帶參數的方法/01 教學演示示例/示例1:帶一個參數的方法/StudentsBiz.java第14章 帶參數的方法/01 教學演示示例/示例1:帶一個參數的方法/TestAdd.java第14章 帶參數的方法/01 教學演示示例/示例2:帶多個參數的方法/StudentsBiz.java第…

java sqlite 工具類_Java 工具類 - JDBC通用操作基類 BaseDao

封裝了增刪改查功能適用于MySQL、Oracle、SQLServer、DB2、Sybase、JTDS、PostgreSql、SQLite、Derby、H2、HSQLDB、ODBC 等等數據庫,有需要的還可以自己增加。package com.tgb.hz.jdbc;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import javax.namin…

java 跨域 下載文件_文件下載重命名(可跨域)

一、正常情況下,我們都如此下載文件并修改文件名,在a標簽上面添加download屬性var link document.createElement(a);link.href file.url;link.download file.name;link.target"_blank";link.click();由于a.download跨域會失效,上…

java hibernate 插入數據_[Java教程]hibernate 返回新插入數據的Id

[Java教程]hibernate 返回新插入數據的Id0 2015-08-28 10:00:11例如 表明 studentInfoString sql"set set nocount on studentInfo(列名,列名) values(值,值);select identity as inserId";java代碼:public int executeCount(String sql, Map paramMap) {…

java輸入行數打印菱形_JAVA題,輸入行數,輸入列數,輸出一個菱形

展開全部1,冒泡排序1. /**2. * JAVA排序算法實現代碼-冒泡(Bubble Sort)排序。3. *4. *5. *6. */7. public class Test {8. public static void main(String[] args) {9. int[] a ;10.11. System.out.print("排序前: ");12.13. for (int i 0; i < a.length; i)1…

mysql 密碼大小寫_MySQL數據庫加密和解密~認證登陸密碼(mysql.user)和MySQL不區分大小寫...

MySQL數據庫認證密碼有兩種方式:1&#xff1a;MySQL 4.1版本之前是MySQL323加密2&#xff1a;MySQL 4.1和之后的版本都是MySQLSHA1加密還有函數:AES_ENCRYPT()加密函數和AES_DECRYPT()解密函數和MD5()加密。MySQL數據庫中自帶old_password(str)和password(str)函數,前者是MySQL…

三星手機 java_如何在三星手機上安裝Java ME應用程序?

我的手機應該可以運行&#xff1a;JavaTM&#xff1a;MIDP 2.0,基于CLDC 1.1的應用程序.但是,無論我嘗試在其上安裝哪個應用程序,我都會收到錯誤&#xff1a;已下載的JAR無效我已經嘗試在Netbeans上構建Java ME項目,使用指定的MIDP 2.0和CLDC 1.1.這些應用程序很簡單,使用Netbe…