13.Redis的过期策略

Redis采用的是定期删除+懒惰删除策略。
(1)定期删除策略
Redis会将每个设置了过期时间的key放入到一个独立的字典里,默认每100ms进行一次过期扫描:
1.随机抽取20个key
2.删除这20个key中过期的key
3.如果过期的key比例超过1/4,重复步骤1,继续删除。
问题1:为什么不扫描所有的key
由于redis是单线程,而且为了防止每次扫描过期的 key 比例都超过 1/4,导致不停循环卡死线程,Redis 为每次扫描添加了上限时间,默认是 25ms。
问题2:为什么 Redis 每次扫描的上限时间是 25ms,还会出现上面的情况?
因为 Redis 是单线程,每个请求处理都需要排队,而且由于 Redis 每次扫描都是 25ms,也就是每个请求最多 25ms,100 个请求就是 2500ms。
如果有大批量的 key 过期,要给过期时间设置一个随机范围,而不宜全部在同一时间过期,分散过期处理的压力。
(2)从库的过期策略
从库不会进行过期扫描,从库对过期的处理是被动的。主库在 key 到期时,会在 AOF 文件里增加一条 del 指令,同步到所有的从库,从库通过执行这条 del 指令来删除过期的 key。
因为指令同步是异步进行的,所以主库过期的 key 的 del 指令没有及时同步到从库的话,会出现主从数据的不一致,主库没有的数据在从库里还存在。
(3)懒惰删除策略
删除指令 del 会直接释放对象的内存,大部分情况下,这个指令非常快,没有明显延迟。不过如果删除的 key 是一个非常大的对象,比如一个包含了千万元素的 hash,又或者在使用 FLUSHDB 和 FLUSHALL 删除包含大量键的数据库时,那么删除操作就会导致单线程卡顿。
redis 4.0 引入了 lazyfree 的机制,它可以将删除键或数据库的操作放在后台线程里执行, 从而尽可能地避免服务器阻塞。
(4)unlink
unlink指令,能对删除操作进行懒处理,丢给后台线程异步回收内存。
(5)flush
flushdb 和 flushall 指令,用来清空数据库,这也是极其缓慢的操作。Redis 4.0 同样给这两个指令也带来了异步化,在指令后面增加 async 参数就可以将操作交给后台线程慢慢进行运行。
(6)异步队列
如果主线程要回收某些大且过期的对象,会将这个key的内存回收操作包装成一个任务,放进异步队列,后台线程会从这个异步队列中取任务,任务队列中,主线程和异步线程可同时操作,因此是一个线程安全的队列。
(7)更多异步删除点
slave-lazy-flush 从库接受完 rdb 文件后的 flush 操作
lazyfree-lazy-eviction 内存达到 maxmemory 时进行淘汰
lazyfree-lazy-expire key 过期删除
lazyfree-lazy-server-del rename 指令删除 destKey
(8)内存淘汰机制
Redis 的内存占用会越来越高。Redis 为了限制最大使用内存,提供了 redis.conf 中的配置参数 maxmemory。当内存超出 maxmemory,Redis 提供了几种内存淘汰机制让用户选择,配置 maxmemory-policy:
1.noeviction:当内存超出 maxmemory,写入请求会报错,但是删除和读请求可以继续。(不推荐)
2.allkeys-lru:当内存超出 maxmemory,在所有的 key 中,移除最少使用的key。只把 Redis 既当缓存是使用这种策略。(推荐)。
3.allkeys-random:当内存超出 maxmemory,在所有的 key 中,随机移除某个 key。
4.volatile-lru:当内存超出 maxmemory,在设置了过期时间 key 的字典中,移除最少使用的 key。把 Redis 既当缓存,又做持久化的时候使用这种策略。
5.volatile-random:当内存超出 maxmemory,在设置了过期时间 key 的字典中,随机移除某个key。
6.volatile-ttl:当内存超出 maxmemory,在设置了过期时间 key 的字典中,优先移除 ttl 小的。
(9)LRU算法
实现 LRU 算法除了需要 key/value 字典外,还需要附加一个链表,链表中的元素按照一定的顺序进行排列。当空间满的时候,会踢掉链表尾部的元素。当字典的某个元素被访问时,它在链表中的位置会被移动到表头。所以链表的元素排列顺序就是元素最近被访问的时间顺序。
Redis使用的是随机LRU算法,Redis 为每一个 key 增加了一个24 bit的字段,用来记录这个 key 最后一次被访问的时间戳。
注意 Redis 的 LRU 淘汰策略是懒惰处理,也就是不会主动执行淘汰策略,当 Redis 执行写操作时,发现内存超出 maxmemory,就会执行 LRU 淘汰算法。这个算法就是随机采样出5(默认值)个 key,然后移除最旧的 key,如果移除后内存还是超出 maxmemory,那就继续随机采样淘汰,直到内存低于 maxmemory 为止。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Redis 过期策略是控制 Redis key 生命周期的重要手段,以下是 Redis 过期策略的知识体系: 1. 过期时间:过期时间是指 Redis key 存在的时间,可以通过 EXPIRE 命令和 PEXPIRE 命令来设置。EXPIRE 命令设置的过期时间是一个固定的时间,而 PEXPIRE 命令设置的过期时间是一个相对时间,即从当前时间开始计算。 2. 过期删除:过期删除是指 Redis 在 key 过期后自动删除 key 的机制。Redis 提供了惰性过期和定期删除两种过期删除策略。 3. 惰性过期:惰性过期是指 Redis 在访问 key 时检查 key 是否过期,如果过期则删除 key。这种策略可以减少 Redis 的负载,但可能会导致过期 key 的数量增多,占用更多的内存空间。 4. 定期删除:定期删除是指 Redis 在每隔一段时间扫描整个数据库,删除过期的 key。这种策略可以减少过期 key 的数量,但可能会导致 Redis 的性能下降。 5. 淘汰策略:当 Redis 内存空间不足时,会触发淘汰策略,即删除一些不常用或者过期的 key,来腾出更多的空间。Redis 提供了多种淘汰策略,例如 LRU(最近最少使用)、LFU(最不经常使用)等。 总之,Redis 过期策略是控制 Redis key 生命周期的重要手段,可以通过过期时间、过期删除、淘汰策略等方式来控制 Redis 中 key 的存储和释放。在面试中,还需要掌握 Redis 过期策略的原理、机制、优缺点、调优等方面的知识。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值