本来想把redis main函数附带都读完再写笔记,但实在太大了,所以现在决定一部分一部分地记录。
eventloop中在每次进入循环时都会调用beforeSleep回调(因为processevents通常都会挂起一定时间等待事件发生),redis的beforeSleep做什么工作呢,看看下面:
void beforeSleep(struct aeEventLoop* eventLoop) {
REDIS_NOTUSED(eventLoop);
listNode* ln;
redisClient* c;
while (listLength(server.unblocked_clients)) {
ln = listFirst(server.unblocked_clients);
redisAssert(ln != NULL);
c = ln->value;
listDelNode(server.unblocked_clients, ln);
c->flags &= ~REDIS_UNBLOCKED;
if (c->querybuf && sdslen(c->querybuf) > 0) {
server.current_client = c;
processInputBuffer(c);
server.current_client = NULL;
}
}
// Write the AOF buffer on disk.
flushAppendOnlyFile(0);
}
主要工作有两步:
1.从unblock_clients链表中拿client,将标志REDIS_UNBLOCKED去掉,然后处理其中的querybuf;
2.flush AOF文件,这个在aof的笔记中有提到。
第二步没什么好说的,第一步中处理querybuf的processInputBuffer大致是这样的:
判断当前命令是否是multi,不是的话调用processInlineBuffer,是的话调用processMultibulkBuffer,这两个函数处理之后,会得到c->argc,c->argv,然后调用processCommand,完了调用resetClient。
如何判断是否是multi呢,就是看缓冲区第一个字节是不是*。
processInlineBuffer简单,它的命令在一行之内,用空格分开,所以容易得到argv和argc;
processMultibulkBuffer中,第一行是个整数,表明有多少个参数(c->argc, c->multibulklen),每个命令参数占一个bulk,一个bulk是由一个指明bulk大小的数字(c->bulklen),bulk本身,还有换行符组成。
待续,还有processCommand呢。