• linkedu视频
  • 平面设计
  • 电脑入门
  • 操作系统
  • 办公应用
  • 电脑硬件
  • 动画设计
  • 3D设计
  • 网页设计
  • CAD设计
  • 影音处理
  • 数据库
  • 程序设计
  • 认证考试
  • 信息管理
  • 信息安全
菜单
linkedu.com
  • 网页制作
  • 数据库
  • 程序设计
  • 操作系统
  • CMS教程
  • 游戏攻略
  • 脚本语言
  • 平面设计
  • 软件教程
  • 网络安全
  • 电脑知识
  • 服务器
  • 视频教程
  • MsSql
  • Mysql
  • oracle
  • MariaDB
  • DB2
  • SQLite
  • PostgreSQL
  • MongoDB
  • Redis
  • Access
  • 数据库其它
  • sybase
  • HBase
您的位置:首页 > 数据库 >Redis > 详解redis是如何实现队列消息的ack

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

作者:Daniel Cai 字体:[增加 减小] 来源:互联网 时间:2017-05-11

Daniel Cai通过本文主要向大家介绍了redis实现消息队列,redis 队列,redis消息队列,php redis队列,java redis队列等相关知识,希望本文的分享对您有所帮助

前言

由于公司提供的队列实在太过于蛋疼而且还限制不能使用其他队列,但为了保证数据安全性需要一个可以有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
</div>

ack时候比较简单,只需要将指定id从set和hash中删除即可

 local key = ARGV[1]
 local qName=ARGV[2]
 redis.call('ZREM', qName..'BACKUP', key)
 redis.call('HDEL', qName..'CONTEXT', key)
</div>

在程序中使用前需要显示load这两个脚本,后面直接调用这两个脚本的sha值即可执行。

总结

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

</div>
分享到:QQ空间新浪微博腾讯微博微信百度贴吧QQ好友复制网址打印

您可能想查找下面的文章:

  • 详解redis是如何实现队列消息的ack
  • Redis实现分布式队列浅析
  • redis实现简单队列
  • Redis实现唯一计数的3种方法分享

相关文章

  • 2017-09-08Redis4基于CentOS6集群搭建
  • 2017-05-11Redis数据库的应用场景介绍
  • 2017-05-11CentOS下Redis数据库的基本安装与配置教程
  • 2017-05-11CentOS系统安装Redis及Redis的PHP扩展详解
  • 2017-05-11Redis Stat的安装指南
  • 2017-05-11在Redis数据库中实现分布式速率限制的方法
  • 2017-05-11基于Redis实现分布式锁以及任务队列
  • 2017-07-23Redis缓存技术学习系列之事务处理
  • 2017-05-11redis常用命令、常见错误、配置技巧等分享
  • 2017-05-11深入了解Redis的性能

文章分类

  • MsSql
  • Mysql
  • oracle
  • MariaDB
  • DB2
  • SQLite
  • PostgreSQL
  • MongoDB
  • Redis
  • Access
  • 数据库其它
  • sybase
  • HBase

最近更新的内容

    • 基于Redis实现分布式锁以及任务队列
    • 利用Redis统计网站在线活跃用户的方法
    • 详解利用redis + lua解决抢红包高并发的问题
    • 详解redis大幅性能提升之使用管道(PipeLine)和批量(Batch)操作
    • Redis持久化之AOF
    • redis常用命令小结
    • 浅谈redis在项目中的应用
    • Redis批量删除KEY的方法
    • 使用Redis实现用户积分排行榜的教程
    • mac下设置redis开机启动方法步骤

关于我们 - 联系我们 - 免责声明 - 网站地图

©2020-2025 All Rights Reserved. linkedu.com 版权所有