国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 數據庫 > Redis > 正文

詳解redis是如何實現隊列消息的ack

2020-03-17 12:38:28
字體:
來源:轉載
供稿:網友

前言

由于公司提供的隊列實在太過于蛋疼而且還限制不能使用其他隊列,但為了保證數據安全性需要一個可以有ack功能的隊列。

原生的redis/252719.html">redis中通過L/R PUSH/POP方式來實現隊列的功能,這個當然是沒辦法滿足需求的(沒有ack功能),所以需要自己對redis的list(隊列)做個小小的調整。

大體思路為在POP時將pop出的數據放到備份的地方,當有ACK請求(確認消息被消耗)后將備份的信息刪除掉;每次在pop前需要檢查備份隊列中有沒有過期的數據沒有ack的,如果有則PUSH到list中后再從list中POP出來。

以下腳本使用lua實現,只需要在執行前加載到redis中即可。

消息本身需要包含id屬性

push沒什么問題,原生即可(此處以LPUSH為例)

pop時腳本

local not_empty = function(x) return (type(x) == "table") and (not x.err) and (#x ~= 0)endlocal qName = ARGV[1] --隊列名稱local currentTime = ARGV[2] --當前時間,這個需要從外部傳入,不能使用redis自身時間,如果使用自身時間可能導致redis本身的backup在重放請求時出現不一致性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]))endtmp = 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)endreturn 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這兩個腳本,后面直接調用這兩個腳本的sha值即可執行。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網的支持。


注:相關教程知識閱讀請移步到Redis頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 锡林郭勒盟| 汽车| 长丰县| 唐山市| 乌苏市| 皋兰县| 革吉县| 青浦区| 余江县| 兴安县| 石柱| 潜江市| 炎陵县| 梅河口市| 平阴县| 绥中县| 株洲县| 西贡区| 平泉县| 崇礼县| 根河市| 修武县| 禄劝| 合作市| 宁强县| 德令哈市| 吉木萨尔县| 象州县| 肥西县| 彝良县| 华容县| 临朐县| 丁青县| 闵行区| 蒙阴县| 安多县| 江西省| 乐昌市| 罗甸县| 灌南县| 双桥区|