redis beforesleep

本来想把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呢。

 

posted @ 2013-06-25 19:49  坚固66  阅读(1161)  评论(1编辑  收藏  举报