• 企业400电话
  • 微网小程序
  • AI电话机器人
  • 电商代运营
  • 全 部 栏 目

    企业400电话 网络优化推广 AI电话机器人 呼叫中心 网站建设 商标✡知产 微网小程序 电商运营 彩铃•短信 增值拓展业务
    详解redis是如何实现队列消息的ack

    前言

    由于公司提供的队列实在太过于蛋疼而且还限制不能使用其他队列,但为了保证数据安全性需要一个可以有ack功能的队列。

    原生的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)
    end
    
    local 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]))
    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这两个脚本,后面直接调用这两个脚本的sha值即可执行。

    总结

    以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

    您可能感兴趣的文章:
    • 详解Redis用链表实现消息队列
    • SpringBoot利用redis集成消息队列的方法
    • PHP使用php-resque库配合Redis实现MQ消息队列的教程
    • Java利用Redis实现消息队列的示例代码
    • phpredis提高消息队列的实时性方法(推荐)
    • PHP基于Redis消息队列实现发布微博的方法
    • php+redis消息队列实现抢购功能
    • 深入理解redis分布式锁和消息队列
    • PHP+Redis 消息队列 实现高并发下注册人数统计的实例
    • redis中队列消息实现应用解耦的方法
    上一篇:Linux中设置Redis开机启动的方法
    下一篇:一起raid数据恢复及回迁成功的案例
  • 相关文章
  • 

    © 2016-2020 巨人网络通讯 版权所有

    《增值电信业务经营许可证》 苏ICP备15040257号-8

    详解redis是如何实现队列消息的ack 详解,redis,是,如何,实现,