一、AOF文件重写的概念
为什么需要AOF文件重写
- 举个例子,如果客户端执行了以下命令:
- 那么服务器为了保存当前list键的状态,必须在AOF文件中写入六条命令
因为AOF持久化是通过保存被执行的写命令来记录数据库状态的,所以随着服务器运行时间的流逝,AOF文件中的内容会越来越多,文件的体积也会越来越大,如果不加以控制的话,体积过大的AOF文件很可能对Redis服务器、甚至整个宿主计算机造成影响,并且AOF文件的体积越大,使用AOF文件来进行数据还原所需的时间就越多
AOF文件重写的原理:为了解决AOF文件体积膨胀的问题,Redis提供了AOF文件重写(rewrite)功能。通过该功能,Redis服务器可以创建一个新的AOF文件来替代现有的AOF文件,新旧两个AOF文件所保存的数据库状态相同,但新AOF文件不会包含任何浪费空间的冗余命令,所以新AOF文件的体积通常会比旧AOF文件的体积要小得多
二、AOF文件重写的实现(aof_rewrite函数)
- 原理:虽然Redis将生成新AOF文件替换旧AOF文件的功能命名为“AOF文件重写”,但实际上, AOF文件重写并不需要对现有的AOF文件进行任何读取、分析或者写入操作,这个功能是通过读取服务器当前的数据库状态来实现的
演示案例: 考虑这样一个情况,如果服务器对list键执行了以下命令:
- 如果不使用AOF文件重写的功能:那么服务器为了保存当前list键的状态,必须在AOF文件中写入六条命令
- 如果使用AOF文件重写的功能:服务不会去读取和分析现有AOF文件的内容,而是 直接从数据库中读取键list的值 ,然后用一条RPUSH list “C” “D” “E” “F”
"G"命令来代替保存在AOF文件中的六条命令,这样就可以将保存list键所需的命令从六条减少为一条
aof_rewrite函数伪代码
- 整个重写过程可以用以下伪代码表示:
def aof_rewrite(new_aof_file_name):
# 创建新 AOF 文件
f = create_file(new_aof_file_name)
# 遍历数据库
for db in redisServer.db:
# 忽略空数据库
if db.is_empty(): continue
# 写入SELECT 命令,指定数据库号码
f.write_command("SELECT" + db.id)
# 遍历数据库中的所有键
for key in db:
# 忽略已过期的键