redis list设置过期时间_面试官:你在Redis中设置过带过期时间的Key吗?
点击上方小伟后端笔记关注公众号
每天阅读Java干货文章
熟悉Redis的同学应该知道,Redis的每个Key都可以设置一个过期时间,当达到过期时间的时候,这个key就会被自动删除。
在为key设置过期时间需要注意的事项
1、 DEL/SET/GETSET等命令会清除过期时间
在使用DEL、SET、GETSET等会覆盖key对应value的命令操作一个设置了过期时间的key的时候,会导致对应的key的过期时间被清除。
//设置mykey的过期时间为300s127.0.0.1:6379> set mykey hello ex 300OK//查看过期时间127.0.0.1:6379> ttl mykey(integer) 294//使用set命令覆盖mykey的内容127.0.0.1:6379> set mykey ollehOK//过期时间被清除127.0.0.1:6379> ttl mykey(integer) -1
2、INCR/LPUSH/HSET等命令则不会清除过期时间
而在使用INCR/LPUSH/HSET这种只是修改一个key的value,而不是覆盖整个value的命令,则不会清除key的过期时间。
INCR:
//设置incr_key的过期时间为300s127.0.0.1:6379> set incr_key 1 ex 300OK127.0.0.1:6379> ttl incr_key(integer) 291//进行自增操作127.0.0.1:6379> incr incr_key(integer) 2127.0.0.1:6379> get incr_key"2"//查询过期时间,发现过期时间没有被清除127.0.0.1:6379> ttl incr_key(integer) 277
LPUSH:
//新增一个list类型的key,并添加一个为1的值127.0.0.1:6379> LPUSH list 1(integer) 1//为list设置300s的过期时间127.0.0.1:6379> expire list 300(integer) 1//查看过期时间127.0.0.1:6379> ttl list(integer) 292//往list里面添加值2127.0.0.1:6379> lpush list 2(integer) 2//查看list的所有值127.0.0.1:6379> lrange list 0 11) "2"2) "1"//能看到往list里面添加值并没有使过期时间清除127.0.0.1:6379> ttl list(integer) 252
3、PERSIST命令会清除过期时间
当使用PERSIST命令将一个设置了过期时间的key转变成一个持久化的key的时候,也会清除过期时间。
127.0.0.1:6379> set persist_key haha ex 300OK127.0.0.1:6379> ttl persist_key(integer) 296//将key变为持久化的127.0.0.1:6379> persist persist_key(integer) 1//过期时间被清除127.0.0.1:6379> ttl persist_key(integer) -1
4、使用RENAME命令,老key的过期时间将会转到新key上
在使用例如:RENAME KEY_A KEY_B命令将KEY_A重命名为KEY_B,不管KEY_B有没有设置过期时间,新的key KEY_B将会继承KEY_A的所有特性。
//设置key_a的过期时间为300s127.0.0.1:6379> set key_a value_a ex 300OK//设置key_b的过期时间为600s127.0.0.1:6379> set key_b value_b ex 600OK127.0.0.1:6379> ttl key_a(integer) 279127.0.0.1:6379> ttl key_b(integer) 591//将key_a重命名为key_b127.0.0.1:6379> rename key_a key_bOK//新的key_b继承了key_a的过期时间127.0.0.1:6379> ttl key_b(integer) 248
这里篇幅有限,我就不一一将key_a重命名到key_b的各个情况列出来,大家可以在自己电脑上试一下key_a设置了过期时间,key_b没设置过期时间这种情况。
5、使用EXPIRE/PEXPIRE设置的过期时间为负数或者使用EXPIREAT/PEXPIREAT设置过期时间戳为过去的时间会导致key被删除
EXPIRE:
127.0.0.1:6379> set key_1 value_1OK127.0.0.1:6379> get key_1"value_1"//设置过期时间为-1127.0.0.1:6379> expire key_1 -1(integer) 1//发现key被删除127.0.0.1:6379> get key_1(nil)
EXPIREAT:
127.0.0.1:6379> set key_2 value_2OK127.0.0.1:6379> get key_2"value_2"//设置的时间戳为过去的时间127.0.0.1:6379> expireat key_2 10000(integer) 1//key被删除127.0.0.1:6379> get key_2(nil)
6、EXPIRE命令可以更新过期时间
对一个已经设置了过期时间的key使用expire命令,可以更新其过期时间。
//设置key_1的过期时间为100s127.0.0.1:6379> set key_1 value_1 ex 100OK127.0.0.1:6379> ttl key_1(integer) 95//更新key_1的过期时间为300s127.0.0.1:6379> expire key_1 300(integer) 1127.0.0.1:6379> ttl key_1(integer) 295
在Redis2.1.3以下的版本中,使用expire命令更新一个已经设置了过期时间的key的过期时间会失败。并且对一个设置了过期时间的key使用LPUSH/HSET等命令修改其value的时候,会导致Redis删除该key。
Redis的过期策略
那你有没有想过一个问题,Redis里面如果有大量的key,怎样才能高效的找出过期的key并将其删除呢,难道是遍历每一个key吗?假如同一时期过期的key非常多,Redis会不会因为一直处理过期事件,而导致读写指令的卡顿。
这里说明一下,Redis是单线程的,所以一些耗时的操作会导致Redis卡顿,比如当Redis数据量特别大的时候,使用keys * 命令列出所有的key。
实际上Redis使用懒惰删除+定期删除相结合的方式处理过期的key。
懒惰删除
所谓懒惰删除就是在客户端访问该key的时候,redis会对key的过期时间进行检查,如果过期了就立即删除。
这种方式看似很完美,在访问的时候检查key的过期时间,不会占用太多的额外CPU资源。但是如果一个key已经过期了,如果长时间没有被访问,那么这个key就会一直存留在内存之中,严重消耗了内存资源。
定期删除
定期删除的原理是,Redis会将所有设置了过期时间的key放入一个字典中,然后每隔一段时间从字典中随机一些key检查过期时间并删除已过期的key。
Redis默认每秒进行10次过期扫描:
从过期字典中随机20个key
删除这20个key中已过期的
如果超过25%的key过期,则重复第一步
同时,为了保证不出现循环过度的情况,Redis还设置了扫描的时间上限,默认不会超过25ms。
作者:千山qianshan
来源:juejin.im/post/5d6bda096fb9a06acc009dc8—END—
推荐阅读:
面试官:为什么 HashMap 的加载因子是0.75?
面试官: 为什么 SQL 语句不要过多的 join?
你还在被 Java NIO 虐?该试试 Netty 了!
新手也能看懂的源码阅读技巧试
如果文章对您有帮助,请您分享、点赞、在看,一波三连支持一下作者,非常感谢!
redis list设置过期时间_面试官:你在Redis中设置过带过期时间的Key吗?相关推荐
- redis查看key的过期时间_面试官:你在Redis中设置过带过期时间的Key吗?
点击上方小伟后端笔记关注公众号 每天阅读Java干货文章 熟悉Redis的同学应该知道,Redis的每个Key都可以设置一个过期时间,当达到过期时间的时候,这个key就会被自动删除. 在为key设置过 ...
- redis怎么修改_面试官问我Redis事务,还问我有哪些实现方式
❝ 「第12期」 距离大叔的80期小目标还有68期,今天大叔要跟大家分享的内容是 -- Reids中的事务.同样,这也是redis中重要指数为四颗星的必备基础知识点.下面一起来了解一下吧. ❞ 相信大 ...
- redis主从复制如何保证数据一致性_面试官:Redis 主从复制时网络开小差了怎么整?...
上周因为实在太忙就认认真真写了一篇水文,吹了一下自己过去的经历,反响竟然超出了我的预期,并且后台还有读者留言表示想看续集的.哈哈,果然大家还是对水文更有热情. 这期我们继续回到之前的 Redis 话题 ...
- redis 删除key的命令_面试官问:Redis变慢了,你会怎么排查?
Redis作为内存数据库,拥有非常高的性能,单个实例的QPS能够达到10W左右.但我们在使用Redis时,经常时不时会出现访问延迟很大的情况,如果你不知道Redis的内部实现原理,在排查问题时就会一头 ...
- java semaphore(0)_面试官:说说Java中的信号量?Semaphore
Semaphore (信号量)是由计算机科学家Dijkstra在1965年提出的,广泛应用不同的操作系统,在管程提出之前信号量就是并发编程领域的霸主!几乎所有并发的语言都支持信号量机制. Semaph ...
- redis查看key的过期时间_面试官:Redis过期后key是怎么样清理的?
前言 笔者一个同事面试某大厂时问到的一个问题,这里拿来讲讲:Redis过期后key是怎么样清理的? 在Redis中,对于过期key的清理主要有惰性清除,定时清理,内存不够时清理三种方法,下面我们就来具 ...
- httpclient 设置超时时间_面试官:技术选型,HttpClient还是OkHttp?
你知道的越多,不知道的就越多,业余的像一棵小草! 你来,我们一起精进!你不来,我和你的竞争对手一起精进! 编辑:业余草 来源:juejin.im/post/6844904040644476941 推荐 ...
- 阅文java面试_面试官:说说Redis的Hash底层 我:......(来自阅文的面试题)
redis源码分析系列文章 前言 hello,各位小可爱们,又见面了.今天这篇文章来自去年面试阅文的面试题,结果被虐了.这一part不说了,下次专门开一篇,写下我面试被虐的名场面,尴尬的不行,全程尬聊 ...
- java 静态方法 调用非静态方法_面试官:为什么java中静态方法不能调用非静态方法或变量?...
这个可能很多人之前学习jvm的时候都会遇到,属于一个小问题,写这篇文章的原因是我在看java相关的面试题目中遇到的,因此顺手总结一下: 一.例子 我们先看效果: 我们在静态方法main中调用非静态变量 ...
最新文章
- UVA 10714 - Ants
- pycharm设置anaconda并运行helloworld
- JSP的9个内置对象-application
- Python入门学习笔记(6)
- 哪些深度相机有python接口_用树莓派和YOLO打造一个深度学习照相机
- mysql对所有id求积_sql 行列式 转换,
- bzoj 3375: [Usaco2004 Mar]Paranoid Cows 发疯的奶牛(二分)
- 通过二层交换机的局域网配置
- 敏捷合同VS传统合同
- iOS视频开发(一):视频采集
- 信息安全-网络安全风险评估技术原理与应用(二)
- 一般的java项目后台都有什么技术?
- 同济大学高等数学第7版笔记和课后答案
- 【已解决】取消电脑自动开机
- 三角函数的极限和导数
- ZKT门禁机标准联接线(按键开关不经过卡机)
- Mac一直要求输入密码怎么办?Mac一直弹输入密码窗口或提示存储钥匙串解决方法
- uniapp 使用map组件 动态自定义标记点图标及内容文字
- 车间调度标准测试集汇总-FJSP、PFSP、JSP、HFSP和分布式车间调度测试集
- 哈工大人工智能暑期课实践项目——手写体识别四则运算(项目计划)
热门文章
- UNIX环境高级编程(三)—— 静态链接库与动态链接库
- 强悍的vim —— 变量的访问
- ubuntu 下 opencv 3. 的安装和运行
- 机器学习基础(八)——感知机(iterative optimization)
- 比大小 log_2^3 与 log_3^5
- 最大似然估计(MLE)的一些公式与定理(python实践)
- Android项目目录结构中各个文件夹的作用
- 2020年python工资一般多少-武汉Python薪资一般是多少?真实数据告诉你
- python 在线培训费用-参加线上python培训班要多少钱?
- 简单python脚本实例-终于晓得python入门脚本实例