@cacheable 设置过期时间_Redis中键过期功能的实现

本文介绍了Redis中键过期功能的实现方式,包括过期时间处理、自动删除过期键策略等。通过设置键的生存时间(TTL),Redis会在指定时间后自动删除过期键。文章详细解释了如何使用SET命令设置过期时间,以及Redis如何通过惰性删除和定期删除策略来管理过期键。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

3bccd47124610b98fc0bab1203806b6a.png

作者:郝赟

前言

最近在需求开发中又用到了我们熟知的Redis字符串操作SET命令,可以设置指定key的值value及该key的生存时间(Time To Live,TTL)。相关命令的语法如下:

set 

这些命令用起来挺熟练,可转念一想,Redis中键的自动过期是如何实现的呢?在翻阅资料及源码的基础上,本文主要从过期时间处理、自动删除过期键策略等方面简要介绍该功能的实现。

键的过期时间处理

设置过期时间

前言中提到,Redis有四个不同命令可以用于设置键的生存时间或过期时间。

可以通过EXPIRE或PEXPIRE命令设置该key的生存时间(Time To Live,TTL),在经过指定秒或毫秒后,Redis服务器就会自动删除生存时间为0的键key。

同时可以使用EXPIREAT或PEXPIREAT命令给键key设置过期时间(expire time)。

虽然命令形式多样,但实际上EXPIRE、PEXPIRE、EXPIREAT三个命令都是使用PEXPIREAT命令来实现的,转换方法很简单,就是将过期时间换算成时间戳,并保持时间单位统一。

保存过期时间

redisDb结构的expires字典保存了数据库中所有键的过期时间,被称为“过期字典”。

  • 过期字典是一个指针,指向键空间中的某个对象(也即是某个数据库键)。
  • 过期字典的值是一个long long类型的整数,保存了键所指向的数据库键的过期时间(一个毫秒精度的UNIX时间戳)。

移除过期时间

PERSIST命令可以移除一个键的过期时间,实际就是PEXPIREAT命令的反操作:PERSIST命令在过期字典中查找给定的键,并解除键和值(过期时间)在过期字典中的关联。

计算并返回剩余生存时间

可以使用TTL或PTTL命令查找给定键key的剩余生存时间(key距离被服务器删除还剩多少秒/毫秒),两个命令都是通过计算键的过期时间和当前时间的差值实现的。

过期键的判定

Redis通过查询过期字典的方式检查一个给定键是否过期:

  1. 检查给定键是否存在于过期字典:如果存在,那么取得键的过期时间;
  2. 检查当前UNIX时间戳是否大于键的过期时间:如果是的话,name键已过期;否则键未过期。

过期键删除策略

常见的三种删除策略对比

8cba9b5a8462ba98c7f1bb1e12b5101f.png

Redis的过期键删除策略

Redis服务器实际使用的是惰性删除和定期删除两种策略:通过配合使用这两种策略,服务器可以很好地在合理使用cpu和避免浪费内存空间之间取得平衡。

惰性删除策略的实现

过期键的惰性删除策略由db.c/expireIfNeeded函数实现,所有读写数据库的Redis命令在执行之前都会调用expireIfNeeded函数对输入键进行检查:

  • 如果输入键已经过期,那么expireIfNeeded函数将输入键从数据库中删除。
  • 如果输入键未过期,那么expireIfNeeded函数不做动作。

expireIfNeeded函数就像一个过滤器,它可以在命令真正执行之前,过滤掉过期的输入键,从而避免命令接触到过期键。另外,因为每个被访问的键都可能因为过期而被expireIfNeeded函数删除,所以每个命令的实现函数都必须能同时处理键存在和不存在的情况:

  • 当键存在时,命令按照键存在的情况执行。
  • 当键不存在或者键因为过期而被expireIfNeeded函数删除时,命令按照键不存在的情况执行。

定期删除策略的实现

过期键的定期删除策略由redis.c/activeExpireCycle函数实现,每当Redis的服务器周期性操作redis.c/serverCron函数执行时,activeExpireCycle函数就会被调用,它在规定的时间内,分多次遍历服务器中的各个数据库,从数据库的expires字典中随机检查一部分键的过期时间,并删除其中的过期键。

activeExpireCycle函数的工作模式可以总结如下:

  • 函数每次运行时,都从一定数量的数据库(取min(默认16,实际数量))中取出一定数量的随机键(默认20)进行检查,并删除其中的过期键。
  • 全局变量current_db会记录当前activeExpireCycle函数检查的进度,并在下一次activeExpireCycle函数调用时,接着上一次的进度进行处理。
  • 随着activeExpireCycle函数的不断执行,服务器中的所有db都会被检查一遍,这时函数将current_db变量重置为0,然后进行新一轮的定期删除。

小结

本文对Redis中键过期功能的实现做了一个简要介绍,相信读者看完之后会对大致的实现方案有所了解,但更多细节推荐阅读《Redis设计与实现》,当然想自己去研究源码更好啦。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值