美文网首页
Redis源码研究之监视器

Redis源码研究之监视器

作者: wenmingxing | 来源:发表于2018-05-07 21:37 被阅读19次

本文主要说明Redis监视器的实现代码。

I、监视器理论部分

1、客户端通过执行MONITOR命令可以将自己变为一个监视器,实时地接受并打印出服务器当前处理的命令请求的相关信息,如:

此时,当其他客户端向服务器发送一条命令请求时,服务器除了会处理这条命令请求之外,还会将这条命令请求的信息发送给所有监视器:

II、代码实现

2.1 成为监视器

redisServer维护一个monitors的链表,记录自己的监视器,每次收到MONITOR命令之后,只需将客户端追加到表尾即可:

/*src/redis.c/monitorCommand*/
void monitorCommand(redisClient *c) {
    /* ignore MONITOR if already slave or in monitor mode */
    if (c->flags & REDIS_SLAVE) return;
        c->flags |= (REDIS_SLAVE|REDIS_MONITOR);
        listAddNodeTail(server.monitors,c);
        addReply(c,shared.ok);  //回复OK
}
2.2 向监视器传播命令

call函数中有对于监视器命令传播的实现:

// call() 函数是执行命令的核心函数,这里只看监视器部分
/*src/redis.c/call*/
/* Call() is the core of Redis execution of a command */
void call(redisClient *c, int flags) {
    long long dirty, start = ustime(), duration;
    int client_old_flags = c->flags;
    /* Sent the command to clients in MONITOR mode, only if the commands are
    * not generated from reading an AOF. */
    if (listLength(server.monitors) &&
        !server.loading &&
        !(c->cmd->flags & REDIS_CMD_SKIP_MONITOR))
        {
        replicationFeedMonitors(c,server.monitors,c->db->id,c->argv,c->argc);
    }
    ......
}  

call主要调用了replicationFeedMonitors,这个函数的作用就是将命令打包为协议,发送给监视器:

// 将协议发给 Monitor
/*src/replication.c/replicationFeedMonitors*/
void replicationFeedMonitors(redisClient *c, list *monitors, int dictid, robj **argv, int argc) {
    listNode *ln;
    listIter li;
    int j;
    sds cmdrepr = sdsnew("+");
    robj *cmdobj;
    struct timeval tv;

    // 获取时间戳
    gettimeofday(&tv,NULL);
    cmdrepr = sdscatprintf(cmdrepr,"%ld.%06ld ",(long)tv.tv_sec,(long)tv.tv_usec);
    if (c->flags & REDIS_LUA_CLIENT) {
        cmdrepr = sdscatprintf(cmdrepr,"[%d lua] ",dictid);
    } else if (c->flags & REDIS_UNIX_SOCKET) {
        cmdrepr = sdscatprintf(cmdrepr,"[%d unix:%s] ",dictid,server.unixsocket);
    } else {
        cmdrepr = sdscatprintf(cmdrepr,"[%d %s] ",dictid,getClientPeerId(c));
    }

    // 获取命令和参数
    for (j = 0; j < argc; j++) {
        if (argv[j]->encoding == REDIS_ENCODING_INT) {
            cmdrepr = sdscatprintf(cmdrepr, "\"%ld\"", (long)argv[j]->ptr);
        } else {
            cmdrepr = sdscatrepr(cmdrepr,(char*)argv[j]->ptr,
                        sdslen(argv[j]->ptr));
        }
        if (j != argc-1)
            cmdrepr = sdscatlen(cmdrepr," ",1);
    }
    cmdrepr = sdscatlen(cmdrepr,"\r\n",2);
    cmdobj = createObject(REDIS_STRING,cmdrepr);

    // 将内容发送给所有 MONITOR 
    listRewind(monitors,&li);
    while((ln = listNext(&li))) {
        redisClient *monitor = ln->value;
        addReply(monitor,cmdobj);
    }
    decrRefCount(cmdobj);
}

【参考】
[1] 《Redis设计与实现》
[2] 《Redis源码日志》

相关文章

  • Redis源码研究之监视器

    本文主要说明Redis监视器的实现代码。 I、监视器理论部分 1、客户端通过执行MONITOR命令可以将自己变为一...

  • Redis 源码研究之dict

    本文主要记录在阅读Redis源码中dict部分的一些函数和实现的巧妙之处。 建议阅读: 1、Redis字典实现的理...

  • Redis源码研究之redisObject

    本文主要说明Redis key-value结构中封装五种value的redisObject结构。 I、上帝视角 r...

  • Redis 源码研究之skiplist

    本文主要记录Redis源码中skiplist数据结构的一些函数实现。 建议阅读: 1、Redis中跳跃表的理论说...

  • Redis 源码研究

    本文集主要记录源码阅读中优秀的部分。 下表中,第一列表示源码所属部分,path表示在Redis中文件的位置, ur...

  • Redis源码研究之哨兵Sentinel

    本文主要说明Redis中哨兵Sentinel的设计与实现。 建议阅读:1、Sentinel的理论部分见:Redis...

  • 关于Redis的一些思考和总结

    日常的业务系统中经常使用到redis,平时也会研究下redis的设计文档和源码,对redis的使用场景、实现方案、...

  • Redis 源码研究之数据淘汰机制

    本文主要介绍Redis的几种数据淘汰机制。 I、上帝视角 由于Redis是内存型数据库,其允许用户设置最大使用内存...

  • Redis源码研究之AOF持久化

    本文主要研究AOF持久化策略的实现方式,了解AOF数据组织方式和运作机制。。 建议阅读: 1、Redis AOF...

  • Redis源码研究之订阅与发布

    本文主要说明Redis的两种订阅模式的实现。 建议阅读: 1、Redis订阅与发布理论说明见:Redis之发布与...

网友评论

      本文标题:Redis源码研究之监视器

      本文链接:https://www.haomeiwen.com/subject/eedprftx.html