詳解redis是如何實現(xiàn)隊列消息的ack
前言
由于公司提供的隊列實在太過而且還限制不能使用其他隊列,但為了保證數(shù)據(jù)安全性需要一個可以有ack功能的隊列。
原生的redis中通過L/R PUSH/POP方式來實現(xiàn)隊列的功能,這個當然是沒辦法滿足需求的(沒有ack功能),所以需要自己對redis的list(隊列)做個小小的調(diào)整。
大體思路為在POP時將pop出的數(shù)據(jù)放到備份的地方,當有ACK請求(確認消息被消耗)后將備份的信息刪除掉;每次在pop前需要檢查備份隊列中有沒有過期的數(shù)據(jù)沒有ack的,如果有則PUSH到list中后再從list中POP出來。
以下腳本使用lua實現(xiàn),只需要在執(zhí)行前加載到redis中即可。
消息本身需要包含id屬性
push沒什么問題,原生即可(此處以LPUSH為例)
pop時腳本
local not_empty = function(x) return (type(x) == "table") and (not x.err) and (#x ~= 0) end local qName = ARGV[1] --隊列名稱 local currentTime = ARGV[2] --當前時間,這個需要從外部傳入,不能使用redis自身時間,如果使用自身時間可能導致redis本身的backup在重放請求時出現(xiàn)不一致性 local considerAsFailMaxTimeSpan = ARGV[3] --超時時間設定,當消息超過一定時間還沒有ack則認為此消息需要再次入隊 local zsetName= qName ..'BACKUP' local hashName= qName ..'CONTEXT' local tmp = redis.call('ZRANGEBYSCORE',zsetName , '-INF', tonumber(currentTime) - tonumber(considerAsFailMaxTimeSpan), 'LIMIT', 0, 1) if (not_empty(tmp)) then redis.call('ZREM', zsetName, tmp[1]) --此處拿出的為消息的唯一id redis.call('LPUSH', qName, redis.call('HGET', hashName, tmp[1])) end tmp = redis.call('RPOP', qName) if (tmp) then local msg = cjson.decode(tmp) local id = msg['id'] redis.call('ZADD', zsetName, tonumber(currentTime), id) redis.call('HSET',hashName , id, tmp) end return tmp
ack時候比較簡單,只需要將指定id從set和hash中刪除即可
local key = ARGV[1] local qName=ARGV[2] redis.call('ZREM', qName..'BACKUP', key) redis.call('HDEL', qName..'CONTEXT', key)
在程序中使用前需要顯示load這兩個腳本,后面直接調(diào)用這兩個腳本的sha值即可執(zhí)行。
總結
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關文章
無法啟動Redis打開redis-server閃退的問題解決辦法
正常開啟redis服務,首先要啟動redis-server.exe,但是閃退,導致無法開啟redis服務,這篇文章主要給大家介紹了關于無法啟動Redis打開redis-server閃退問題的解決辦法,需要的朋友可以參考下2024-07-07redis緩存數(shù)據(jù)庫中數(shù)據(jù)的方法
這篇文章主要為大家詳細介紹了redis緩存數(shù)據(jù)庫中數(shù)據(jù)的方法,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-07-07