作者:罗曼蒂克
链接:https://www.jianshu.com/p/137aab0b0a1b
有没有想过Redis中过期的那些键去哪了?是谁在什么时候怎么删掉的?
先来介绍一下各种方案:
在这三种策略中,第一种和第三种为主动删除策略,而第二种则为被动删除策略。
★Redis服务器实际使用的是惰性删除和定期删除两种策略:通过配合使用这两种删除策略,服务器可以很好地在合理使用CPU时间和避免浪费内存空间之间取得平衡。 ”
过期键的惰性删除策略由db.c/expireIfNeeded函数实现,所有读写数据库的Redis命令在执行之前都会调用expireIfNeeded函数对输入键进行检查:
expireIfNeeded函数就像一个过滤器,它可以在命令真正执行之前,过滤掉过期的输入键,从而避免命令接触到过期键。
另外,因为每个被访问的键都可能因为过期而被expireIfNeeded函数删除,所以每个命令的实现函数都必须能同时处理键存在以及键不存在这两种情况:
举个例子,下图展示了GET命令的执行过程,在这个执行过程中,命令需要判断键是否存在以及键是否过期,然后根据判断来执行合适的动作。
过期键的定期删除策略由redis.c/activeExpireCycle函数实现,每当Redis的服务器周期性操作redis.c/serverCron函数执行时,activeExpireCycle函数就会被调用,它在规定的时间内,分多次遍历服务器中的各个数据库,从数据库的expires字典中随机检查一部分键的过期时间,并删除其中的过期键。
DEFAULT_DB_NUMBERS = 16 //默认每次检查的数据库数量
DEFAULT_KEY_NUMBERS = 20 //默认每个数据库检查的键数量
current_db = 0 //全局变量,记录检查进度
//以实际数据库数量为准
if server.dbnum < DEFAULT_DB_NUMBERS:
db_numbers = server.dbnum
else:
db_numbers = DEFAULT_DB_NUMBERS
//遍历各个数据库
for i in range(db_numbers):
if current_db == server.dbnum://如果current_db的值等于服务器的数据库数量,
current_db = 0 //这表示检查程序已经遍历了服务器的所有数据库一次,将current_db重置为0,开始新的一轮遍历
redisDb = server.db[current_db] //获取当前要处理的数据库
current_db += 1 //将数据库索引增1,指向下一个要处理的数据库
for j in range(DEFAULT_KEY_NUMBERS):
if redisDb.expires.size() == 0: break //如果数据库中没有一个键带有过期时间,那么跳过这个数据库
key_with_ttl = redisDb.expires.get_random_key() //随机获取一个带有过期时间的键
if is_expired(key_with_ttl)://检查键是否过期,如果过期就删除它
delete_key(key_with_ttl)
if reach_time_limit(): return//已达到时间上限,停止处理</code>
activeExpireCycle函数的工作模式可以总结如下:
本文参考自 《Redis设计与实现》