因为Redis数据是基于内存读写,为防止Redis服务器关闭或者宕机造成数据的丢失,我们通常需要对redis做持久化,即:把内存中的数据(命令)保存一份到磁盘中来做一份备份,当redis服务关闭或宕机后,在Redis服务器重启后将数据从磁盘加载到内存中,不至于造成数据的丢失。
Redis提供了两种持久化方式RDB
和AOF
,可以通过修改redis.conf
来配置,开启Redis持久化后,在redis的安装目录会看到持久化文件:“appendonly.aof
”和“dump.rdb
”
Redis持久化原理
RDB持久化是把当前进程数据生成快照保存到磁盘中的过程
,快照文件就是一个RDB文件(二进制),当我们需要对Redis的数据进行恢复时,我们就可以去加载这个文件,将某时刻的备份文件恢复到redis中。
手动触发:
阻塞当前redis服务器
,直到RDB完成为止,阻塞耗时较长,一般不在线上环境使用RDB的持久化由子进程负责
,完成后自动结束,阻塞只发生在fork阶段,一般时间比较短自动触发:
bgsave
save 900 1 # 当在900秒改变了1条数据,将触发 bgsave 操作
save 300 10 # 当在300秒改变了10条数据,将触发 bgsave 操作
save 60 10000 # 当在60秒改变了10000条数据,将触发 bgsave 操作
dbfilename dump-${port}.rdb # 一般的文件命名方式,加上端口区分
dir /bigdiskpath # 选择一个比较大的硬盘路径,甚至分盘
stop-writes-on-bgsave-error yes # 如果 bgsave 发生了错误,就停止写入
rdbcompression yes # 一般采用压缩格式
rdbchecksum yes # 是否采用校验和的方式,一般采用
备份
,全量复制
等场景以独立日志的方式记录每次写命令
,重启的时候在执行AOF文件中的命令达到恢复数据的目的,AOF的主要作用是解决持久化的实时性
AOF 的工作流程:命令写入(append)
、文件同步(sync)
、文件重写(rewrite)
、重启加载(load)
随着命令不断的写入AOF,文件会越来越大,于是引入了AOF重写机制来压缩文件的体积,本质就是把一些过期的、没有用的、重复的、以及一些可以优化的命令进行一个简化
,简化成一个较小的AOF文件。
AOF重写的两种方式:
bgrewriteaof
:客户端输入 bgwriteaof 命令,redis 会使用一个 linux 的 fork() 函数,生成一个子进程,子进程再进行 AOF 重写生成新的 AOF 文件。自动触发
# 配置
auto-aof-rewrite-min-size # AOF文件重写需要的尺寸,也就是说当AOF多大的时候,才进行AOF的重写
auto-aof-rewrite-percentage # AOF文件增长率,比如这一次重写到了100M,下一次多久重写呢,假如说达到200M就重写,那么增长率就是100%
# 统计项
aof_current_size # AOF当前尺寸,aof_current_size > auto-aof-rewrite-min-size 触发重写
aof_base_size # AOF上次启动和重写的尺寸,增长率 = (aof_current_size - aof_base_size) / aof_base_size,增长率 > auto-aof-rewrite-percentage 触发重写
重写流程
# AOF配置
appendonly yes # 要使用AOF所有功能就置为 yes,默认是 no
appendfilename "appendonly-${port}.aof" # AOF文件名称
appendfsync everysec # 每秒进行刷盘策略
dir /bigdiskpath # 保存目录
no-appendfsync-on-rewrite yes # 是否要做AOF的append操作,yes表示不做这个操作
# AOF重写配置
auto-aof-rewrite-min-size 64MB # AOF文件重写需要的尺寸,也就是说当AOF多大的时候,才进行AOF的重写
auto-aof-rewrite-percentage 100 # AOF文件增长率,比如这一次重写到了100M,下一次多久重写呢,假如说达到200M就重写,那么增长率就是100%
重启 Redis 时,我们很少使用 RDB来恢复内存状态,因为会丢失大量数据。我们通常使用 AOF 日志重放,但是重放 AOF 日志性能相对 RDB来说要慢很多,这样在 Redis 实例很大的情况下,启动需要花费很长的时间
。 Redis 4.0 为了解决这个问题,带来了一个新的持久化选项——混合持久化
。
通过如下配置可以开启混合持久化(必须先开启aof):
aof‐use‐rdb‐preamble yes
如果开启了混合持久化,AOF在重写时,不再是单纯将内存数据转换为RESP命令写入AOF文件,而是将重写这一刻之前的内存做RDB快照处理,并且将RDB快照内容和增量的AOF修改内存数据的命令存在一起,都写入新的AOF文件,新的文件一开始不叫appendonly.aof,等到重写完新的AOF文件才会进行改名,覆盖原有的AOF文件,完成新旧两个AOF文件的替换。
于是在 Redis 重启的时候,可以先加载 RDB 的内容,然后再重放增量 AOF 日志就可以完全替代之前的AOF 全量文件重放,因此重启效率大幅得到提升。
fork 操作
改善 fork 操作的耗时
:
AOF 追加阻塞
当开启 AOF 持久化时,常用的同步硬盘的策略是
everysec
,用于平衡性能和数据安全性。对于这种方式,Redis 使用另一条线程每秒执行 fsync 同步硬盘。当系统硬盘资源繁忙时,会造成 Redis 主线程阻塞。
塞流程分析