INCR key

起始版本:1.0.0

时间复杂度:O(1)

http://www.redis.cn/map.html   (redis知识树)

作   用

对存储在指定key的数值执行原子的加1操作。

如果指定的key不存在,那么在执行incr操作之前,会先将它的值设定为0

如果指定的key中存储的值不是字符串类型(fix:)或者存储的字符串类型不能表示为一个整数,

那么执行这个命令时服务器会返回一个错误(eq:(error) ERR value is not an integer or out of range)。

这个操作仅限于64位的有符号整型数据。

注意: 由于redis并没有一个明确的类型来表示整型数据,所以这个操作是一个字符串操作。

执行这个操作的时候,key对应存储的字符串被解析为10进制的64位有符号整型数据

事实上,Redis 内部采用整数形式(Integer representation)来存储对应的整数值,所以对该类字符串值实际上是用整数保存,也就不存在存储整数的字符串表示(String representation)所带来的额外消耗。

返回值

integer-reply:执行递增操作后key对应的值。

例子

redis> SET mykey "10"
OK
redis> INCR mykey
(integer) 11
redis> GET mykey
"11"
redis>

实例:计数器

Redis的原子递增操作最常用的使用场景是计数器。

使用思路是:每次有相关操作的时候,就向Redis服务器发送一个incr命令。

例如这样一个场景:我们有一个web应用,我们想记录每个用户每天访问这个网站的次数。

web应用只需要通过拼接用户id和代表当前时间的字符串作为key,每次用户访问这个页面的时候对这个key执行一下incr命令。

这个场景可以有很多种扩展方法:

  • 通过结合使用INCR和EXPIRE命令,可以实现一个只记录用户在指定间隔时间内的访问次数的计数器
  • 客户端可以通过GETSET命令获取当前计数器的值并且重置为0
  • 通过类似于DECR或者INCRBY等原子递增/递减的命令,可以根据用户的操作来增加或者减少某些值比如在线游戏,需要对用户的游戏分数进行实时控制,分数可能增加也可能减少。

实例: 限速器

限速器是一种可以限制某些操作执行速率的特殊场景。

传统的例子就是限制某个公共api的请求数目。

假设我们要解决如下问题:限制某个api每秒每个ip的请求次数不超过10次。

我们可以通过incr命令来实现两种方法解决这个问题。

实例: 限速器 1

更加简单和直接的实现如下:

FUNCTION LIMIT_API_CALL(ip)
ts = CURRENT_UNIX_TIME()
keyname = ip+":"+ts
current = GET(keyname)
IF current != NULL AND current > 10 THENERROR "too many requests per second"
ELSEMULTIINCR(keyname,1)EXPIRE(keyname,10)EXECPERFORM_API_CALL()
END

这种方法的基本点是每个ip每秒生成一个可以记录请求数的计数器。

但是这些计数器每次递增的时候都设置了10秒的过期时间,这样在进入下一秒之后,redis会自动删除前一秒的计数器。

注意上面伪代码中我们用到了MULTI和EXEC命令,将递增操作和设置过期时间的操作放在了一个事务中,从而保证了两个操作的原子性。

实例: 限速器 2

另外一个实现是对每个ip只用一个单独的计数器(不是每秒生成一个),但是需要注意避免竟态条件。我们会对多种不同的变量进行测试。

FUNCTION LIMIT_API_CALL(ip):
current = GET(ip)
IF current != NULL AND current > 10 THENERROR "too many requests per second"
ELSEvalue = INCR(ip)IF value == 1 THENEXPIRE(value,1)ENDPERFORM_API_CALL()
END

上述方法的思路是,从第一个请求开始设置过期时间为1秒。如果1秒内请求数超过了10个,那么会抛异常。

否则,计数器会清零。

上述代码中,可能会进入竞态条件,比如客户端在执行INCR之后,没有成功设置EXPIRE时间。这个ip的key会造成内存泄漏,直到下次有同一个ip发送相同的请求过来。

把上述INCR和EXPIRE命令写在lua脚本并执行EVAL命令可以避免上述问题(只有redis版本>=2.6才可以使用)

local current
current = redis.call("incr",KEYS[1])
if tonumber(current) == 1 thenredis.call("expire",KEYS[1],1)
end

还可以通过使用redis的list来解决上述问题避免进入竞态条件。

实现代码更加复杂并且利用了一些redis的新的feature,可以记录当前请求的客户端ip地址。这个有没有好处取决于应用程序本身。

FUNCTION LIMIT_API_CALL(ip)
current = LLEN(ip)
IF current > 10 THENERROR "too many requests per second"
ELSEIF EXISTS(ip) == FALSEMULTIRPUSH(ip,ip)EXPIRE(ip,1)EXECELSERPUSHX(ip,ip)ENDPERFORM_API_CALL()
END

The RPUSHX command only pushes the element if the key already exists.

RPUSHX命令会往list中插入一个元素,如果key存在的话

上述实现也可能会出现竞态,比如我们在执行EXISTS指令之后返回了false,但是另外一个客户端创建了这个key。

后果就是我们会少记录一个请求。但是这种情况很少出现,所以我们的请求限速器还是能够运行良好的。

redis---incr命令相关推荐

  1. php redis获取incr的值,Redis Incr命令

    Redis Incr命令 Redis Incr 命令将 key 中储存的数字值增一. 如果 key 不存在,那么 key 的值会先被初始化为 0 ,然后再执行 INCR 操作. 如果值包含错误的类型, ...

  2. Redis INCR命令

    路人甲:嘿,兄弟,知不知道redis的incr命令怎么用? 路人丙:啥?你这都不知道,不就是将key值增1嘛? 路人甲:可以一直一直一直加吗? 路人丙:-- 下图是Redis命令参考网站给出的incr ...

  3. redis incr命令最大值问题

    redis incr 最大值是 Long.MAX_VALUE的值 大于long.MAX_VALUE后就会抛出异常 @Testpublic void test3() throws Interrupted ...

  4. Redis爬坑记(一):incr命令和expire命令的误区

    关注公众号 要实现的功能:限制用户的每分钟的访问次数一个有严重bug的代码:每次访问来了,就执行代码块二,当第一次访问,就走else语句,设置当前用户的次数为1,且设置该key的有效期是一分钟. 在一 ...

  5. redis中的incr命令和incrby命令

    Redis Incr 命令将 key 中储存的数字值增一,如果 key 不存在,那么 key 的值会先被初始化为 0 ,然后再执行 INCR 操作. Redis Incrby 命令将 key 中储存的 ...

  6. java使用redis incr,JFinal Redis plugin 有关数值类型incr操作的bug

    先上测试代码:public class TestCache { private static final Cache testCache = Redis.use(CacheCons.TEST) ; p ...

  7. redis incr和incrBy的使用

     最近用incr和incrBy在接口里做了下埋点统计每天请求总数,这两个命令还是挺好用的,先说下这俩命令吧 注:redis后台服务是串行的单线程执行,不存在并发,即多线程调用Incr/incrby方法 ...

  8. 使用redis incr处理并发,存在死锁问题

    目录 1.项目场景: 2.问题代码 : 3.修改后的代码: 4.实战示例: 5.更新:setIfAbsent 6.再次更新:redisson 1.项目场景: 锁主要是用来实现资源共享同步,只有获取到了 ...

  9. 使用redis incr处理并发问题

    一.背景 最近公司某个短信接口因为没有加验证码限制被恶意调用,最好的解决办法是做在发送短信请求前做一个验证码验证通过后再调用短信接口.但是由于需要立马改动,借此使用了一下 "缓兵之计&quo ...

  10. Redis Incr和ncrby

    Redis Incr 命令将 key 中储存的数字值增1,如果 key 不存在,那么 key 的值会先被初始化为 0 ,然后再执行 INCR 操作. Redis Incrby 命令将 key 中储存的 ...

最新文章

  1. Android移动开发之【Android实战项目】Recyclerview添加花色分割线
  2. Linux chmod命令修改文件与文件夹权限的命令附实例
  3. 【深度学习】CVPR 2021 全部论文链接公布!最新1660篇论文合集!附下载链接
  4. STL 之adjacent_find, merge,inplace_merge
  5. grav html5,如何通过html5实现摇一摇的功能
  6. HTML5摩托车配件销售商城模板
  7. python全局变量怎么删除_python 全局变量怎么改
  8. 超经典两万字,MySQL数据库快速入门。
  9. 有没有测试颜色的软件,用什么软件测试显示器色彩最准:色彩校正软件
  10. 使用frida获取微信EnMicroMsg.db 数据库密码
  11. 电脑编程就业找哪方面
  12. LoRa网关市场现状及未来发展趋势
  13. python+keras实现语音识别
  14. RabbitMmq基础(三)入门应用
  15. GPS时间转北京时间
  16. Android实现横竖屏切换,以及手机横竖屏旋转的时候,实现横竖屏切换
  17. 华三交换机 流量镜像
  18. video读取服务器本地的视频文件,javascript-video.js无法读取本地视频
  19. 智慧社区综合信息服务平台,让你的社区更智能
  20. 如何提升测试人员在公司的地位

热门文章

  1. 常用的函数:atoi,itoa,atof,_ttoi等
  2. Hashtable的用法
  3. 基于B/S的校园餐厅网上订餐系统
  4. 单目摄像头检测输出 3D 边界框
  5. java jacob api_jacob使用入门 | 学步园
  6. Xshell连接ubuntu后vi编辑器中数字小键盘乱码
  7. 对称矩阵的三对角分解(Lanzos分解算法)-MINRES算法预热
  8. workon 未找到命令
  9. CSS选择器(select)
  10. SpringBoot 导出 PDF 图表(折现图、饼状图等)