You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

186 lines
15 KiB
Markdown

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

# 19 | AOF重写触发时机与重写的影响
你好,我是蒋德钧。
我们知道Redis除了使用内存快照RDB来保证数据可靠性之外还可以使用AOF日志。不过RDB文件是将某一时刻的内存数据保存成一个文件而AOF日志则会记录接收到的所有写操作。如果Redis server的写请求很多那么AOF日志中记录的操作也会越来越多进而就导致AOF日志文件越来越大。
所以为了避免产生过大的AOF日志文件Redis会对AOF文件进行重写也就是针对当前数据库中每个键值对的最新内容记录它的插入操作而不再记录它的历史写操作了。这样一来重写后的AOF日志文件就能变小了。
**那么AOF重写在哪些时候会被触发呢以及AOF重写需要写文件这个过程会阻塞Redis的主线程进而影响Redis的性能吗**
今天这节课我就来给你介绍下AOF重写的代码实现过程通过了解它的代码实现我们就可以清楚地了解到AOF重写过程的表现以及它对Redis server的影响。这样当你再遇到Redis server性能变慢的问题时你就可以排查是否是AOF重写导致的了。
接下来我们先来看下AOF重写函数以及它的触发时机。
## AOF重写函数与触发时机
首先实现AOF重写的函数是**rewriteAppendOnlyFileBackground**,它是在[aof.c](https://github.com/redis/redis/tree/5.0/src/aof.c)文件中实现的。在这个函数中会调用fork函数创建一个AOF重写子进程来实际执行重写操作。关于这个函数的具体实现我稍后会给你详细介绍。这里呢我们先来看看这个函数会被哪些函数调用这样我们就可以了解AOF重写的触发时机了。
实际上rewriteAppendOnlyFileBackground函数一共会在三个函数中被调用。
**第一个是bgrewriteaofCommand函数。**这个函数是在aof.c文件中实现的对应了我们在Redis server上执行bgrewriteaof命令也就是说我们手动触发了AOF rewrite的执行。
不过即使我们手动执行了bgrewriteaof命令bgrewriteaofCommand函数也会根据以下两个条件来判断是否实际执行AOF重写。
* **条件一当前是否已经有AOF重写的子进程正在执行。**如果有的话那么bgrewriteaofCommand函数就不再执行AOF重写了。
* **条件二当前是否有创建RDB的子进程正在执行。**如果有的话bgrewriteaofCommand函数会把全局变量server的aof\_rewrite\_scheduled成员变量设置为1这个标志表明Redis server已经将AOF重写设为待调度运行等后续条件满足时它就会实际执行AOF重写我们一会儿就会看到当aof\_rewrite\_scheduled设置为1以后Redis server会在哪些条件下实际执行重写操作
所以这也就是说只有当前既没有AOF重写子进程也没有RDB子进程bgrewriteaofCommand函数才会立即调用rewriteAppendOnlyFileBackground函数实际执行AOF重写。
以下代码展示了bgrewriteaofCommand函数的基本执行逻辑你可以看下。
```
void bgrewriteaofCommand(client *c) {
if (server.aof_child_pid != -1) {
.. //有AOF重写子进程因此不执行重写
} else if (server.rdb_child_pid != -1) {
server.aof_rewrite_scheduled = 1; //有RDB子进程将AOF重写设置为待调度运行
...
} else if (rewriteAppendOnlyFileBackground() == C_OK) { //实际执行AOF重写
...
}
...
}
```
**第二个是startAppendOnly函数。**这个函数也是在aof.c文件中实现的它本身会被configSetCommand函数在[config.c](https://github.com/redis/redis/tree/5.0/src/config.c)文件中和restartAOFAfterSYNC函数在[replication.c](https://github.com/redis/redis/tree/5.0/src/replication.c)文件中)调用。
首先对于configSetCommand函数来说它对应了我们在Redis中执行config命令启用AOF功能如下所示
```
config set appendonly yes
```
这样一旦AOF功能启用后configSetCommand函数就会调用startAppendOnly函数执行一次AOF重写。
而对于restartAOFAfterSYNC函数来说它会在主从节点的复制过程中被调用。简单来说就是当主从节点在进行复制时如果从节点的AOF选项被打开那么在加载解析RDB文件时AOF选项就会被关闭。然后无论从节点是否成功加载了RDB文件restartAOFAfterSYNC函数都会被调用用来恢复被关闭的AOF功能。
那么在这个过程中restartAOFAfterSYNC函数就会调用startAppendOnly函数并进一步调用rewriteAppendOnlyFileBackground函数来执行一次AOF重写。
这里你要注意和bgrewriteaofCommand函数类似**startAppendOnly函数也会判断当前是否有RDB子进程在执行**如果有的话它会将AOF重写设置为待调度执行。除此之外如果startAppendOnly函数检测到有AOF重写子进程在执行那么它就会把该子进程先kill掉然后再调用rewriteAppendOnlyFileBackground函数进行AOF重写。
所以到这里我们其实可以发现无论是bgrewriteaofCommand函数还是startAppendOnly函数当它们检测到有RDB子进程在执行的时候就会把aof\_rewrite\_scheduled变量设置为1这表示AOF重写操作将在条件满足时再被执行。
**那么Redis server什么时候会再检查AOF重写操作的条件是否满足呢**这就和rewriteAppendOnlyFileBackground函数被调用的第三个函数serverCron函数相关了。
**第三个是serverCron函数。**在Redis server运行时serverCron函数是会被周期性执行的。然后它在执行的过程中会做两次判断来决定是否执行AOF重写。
首先serverCron函数会检测当前是否**没有RDB子进程和AOF重写子进程在执行**,并检测是否**有AOF重写操作被设置为了待调度执行**也就是aof\_rewrite\_scheduled变量值为1。
如果这三个条件都满足那么serverCron函数就会调用rewriteAppendOnlyFileBackground函数来执行AOF重写。serverCron函数里面的这部分执行逻辑如下所示
```
//如果没有RDB子进程也没有AOF重写子进程并且AOF重写被设置为待调度执行那么调用rewriteAppendOnlyFileBackground函数进行AOF重写
if (server.rdb_child_pid == -1 && server.aof_child_pid == -1 &&
server.aof_rewrite_scheduled)
{
rewriteAppendOnlyFileBackground();
}
```
事实上这里的代码也回答了我们刚才提到的问题待调度执行的AOF重写会在什么时候执行
其实如果AOF重写没法立即执行的话我们也不用担心。因为**只要aof\_rewrite\_scheduled变量被设置为1了那么serverCron函数就默认会每100毫秒执行并检测这个变量值**。所以如果正在执行的RDB子进程和AOF重写子进程结束了之后被调度执行的AOF重写就可以很快得到执行。
其次即使AOF重写操作没有被设置为待调度执行serverCron函数也会**周期性判断是否需要执行AOF重写**。这里的判断条件主要有三个分别是AOF功能已启用、AOF文件大小比例超出阈值以及AOF文件大小绝对值超出阈值。
这样一来当这三个条件都满足时并且也没有RDB子进程和AOF子进程在运行的话此时serverCron函数就会调用rewriteAppendOnlyFileBackground函数执行AOF重写。这部分的代码逻辑如下所示
```
//如果AOF功能启用、没有RDB子进程和AOF重写子进程在执行、AOF文件大小比例设定了阈值以及AOF文件大小绝对值超出了阈值那么进一步判断AOF文件大小比例是否超出阈值
if (server.aof_state == AOF_ON && server.rdb_child_pid == -1 && server.aof_child_pid == -1 && server.aof_rewrite_perc && server.aof_current_size > server.aof_rewrite_min_size) {
//计算AOF文件当前大小超出基础大小的比例
long long base = server.aof_rewrite_base_size ? server.aof_rewrite_base_size : 1;
long long growth = (server.aof_current_size*100/base) - 100;
//如果AOF文件当前大小超出基础大小的比例已经超出预设阈值那么执行AOF重写
if (growth >= server.aof_rewrite_perc) {
...
rewriteAppendOnlyFileBackground();
}
}
```
那么从这里的代码中你会看到为了避免AOF文件过大导致占用过多的磁盘空间以及增加恢复时长你其实可以通过设置redis.conf文件中的以下两个阈值来让Redis server自动重写AOF文件。
* **auto-aof-rewrite-percentage**AOF文件大小超出基础大小的比例默认值为100%即超出1倍大小。
* **auto-aof-rewrite-min-size**AOF文件大小绝对值的最小值默认为64MB。
好了到这里我们就了解了AOF重写的四个触发时机这里我也给你总结下方便你回顾复习。
* 时机一bgrewriteaof命令被执行。
* 时机二主从复制完成RDB文件解析和加载无论是否成功
* 时机三AOF重写被设置为待调度执行。
* 时机四AOF被启用同时AOF文件的大小比例超出阈值以及AOF文件的大小绝对值超出阈值。
另外这里你还需要注意在这四个时机下其实都不能有正在执行的RDB子进程和AOF重写子进程否则的话AOF重写就无法执行了。
所以接下来我们就来学习下AOF重写的基本执行过程。
## AOF重写的基本过程
首先我们再来看下刚才介绍的rewriteAppendOnlyFileBackground函数。这个函数的主体逻辑比较简单一方面它会通过调用fork函数创建一个子进程然后在子进程中调用rewriteAppendOnlyFile函数进行AOF文件重写。
rewriteAppendOnlyFile函数是在aof.c文件中实现的。它主要会调用**rewriteAppendOnlyFileRio函数**在aof.c文件中来完成AOF日志文件的重写。具体来说就是rewriteAppendOnlyFileRio函数会遍历Redis server的每一个数据库把其中的每个键值对读取出来然后记录该键值对类型对应的插入命令以及键值对本身的内容。
比如如果读取的是一个String类型的键值对那么rewriteAppendOnlyFileRio函数就会记录SET命令和键值对本身内容而如果读取的是Set类型键值对那么它会记录SADD命令和键值对内容。这样一来当需要恢复Redis数据库时我们重新执行一遍AOF重写日志中记录的命令操作就可以依次插入所有键值对了。
另一方面在父进程中这个rewriteAppendOnlyFileBackground函数会**把aof\_rewrite\_scheduled变量设置为0**同时记录AOF重写开始的时间以及记录AOF子进程的进程号。
此外rewriteAppendOnlyFileBackground函数还会调用**updateDictResizePolicy函数**禁止在AOF重写期间进行rehash操作。这是因为rehash操作会带来较多的数据移动操作对于AOF重写子进程来说这就意味着父进程中的内存修改会比较多。因此AOF重写子进程就需要执行更多的写时复制进而完成AOF文件的写入这就会给Redis系统的性能造成负面影响。
以下代码就展示了rewriteAppendOnlyFileBackground函数的基本执行逻辑你可以看下。
```
int rewriteAppendOnlyFileBackground(void) {
...
if ((childpid = fork()) == 0) { //创建子进程
...
//子进程调用rewriteAppendOnlyFile进行AOF重写
if (rewriteAppendOnlyFile(tmpfile) == C_OK) {
size_t private_dirty = zmalloc_get_private_dirty(-1);
...
exitFromChild(0);
} else {
exitFromChild(1);
}
}
else{ //父进程执行的逻辑
...
server.aof_rewrite_scheduled = 0;
server.aof_rewrite_time_start = time(NULL);
server.aof_child_pid = childpid; //记录重写子进程的进程号
updateDictResizePolicy(); //关闭rehash功能
}
```
而从这里你可以看到AOF重写和RDB创建是比较类似的它们都会创建一个子进程来遍历所有的数据库并把数据库中的每个键值对记录到文件中。不过AOF重写和RDB文件又有两个不同的地方
* 一是AOF文件中是以“命令+键值对”的形式来记录每个键值对的插入操作而RDB文件记录的是键值对数据本身
* 二是在AOF重写或是创建RDB的过程中主进程仍然可以接收客户端写请求。不过因为RDB文件只需要记录某个时刻下数据库的所有数据就行而AOF重写则需要尽可能地把主进程收到的写操作也记录到重写的日志文件中。所以AOF重写子进程就需要有相应的机制来和主进程进行通信以此来接收主进程收到的写操作。
下图就展示了rewriteAppendOnlyFileBackground函数执行的基本逻辑、主进程和AOF重写子进程各自执行的内容以及主进程和子进程间的通信过程你可以再来整体回顾下。
![](https://static001.geekbang.org/resource/image/01/dd/01ce2381652fee284c081f7a376006dd.jpg?wh=2000x1125)
到这里我们就大概掌握了AOF重写的基本执行过程。但是在这里你可能还会有疑问比如说AOF重写的子进程和父进程它们之间的通信过程是怎么样的呢
其实,这个通信过程是通过操作系统的**管道机制**pipe来实现的不过你也别着急这部分内容我会在下一讲给你详细介绍。
## 小结
今天这节课我给你介绍了Redis AOF重写机制的实现你需要重点关注以下两个要点
* **AOF重写的触发时机**。这既包括了我们主动执行bgrewriteaof命令也包括了Redis server根据AOF文件大小而自动触发的重写。此外在主从复制的过程中从节点也会启动AOF重写形成一份完整的AOF日志以便后续进行恢复。当然你也要知道当要触发AOF重写时Redis server是不能运行RDB子进程和AOF重写子进程的。
* **AOF重写的基本执行过程**。AOF重写和RDB创建的过程类似它也是创建了一个子进程来完成重写工作。这是因为AOF重写操作实际上需要遍历Redis server上的所有数据库把每个键值对以插入操作的形式写入日志文件而日志文件又要进行写盘操作。所以Redis源码使用子进程来实现AOF重写这就避免了阻塞主线程也减少了对Redis整体性能的影响。
不过你需要注意的是虽然AOF重写和RDB创建都用了子进程但是它们也有不同的地方AOF重写过程中父进程收到的写操作也需要尽量写入AOF重写日志在这里Redis源码是使用了**管道机制**来实现父进程和AOF重写子进程间的通信的。在下一讲中我就会重点给你介绍Redis是如何使用管道完成父子进程的通信以及它们通过管道又传递了哪些数据或信息。
## 每课一问
RDB文件的创建是由一个子进程来完成的而AOF重写也是由一个子进程完成的这两个子进程可以各自单独运行。那么请你思考一下为什么Redis源码中在有RDB子进程运行时不会启动AOF重写子进程呢