设为首页 收藏本站
查看: 1047|回复: 0

[经验分享] redis源码阅读笔记(11)——服务器与客户端

[复制链接]

尚未签到

发表于 2016-12-21 06:52:32 | 显示全部楼层 |阅读模式
1.高层视角
可首先阅读《Redis设计与实现》中的服务器与客户端
原文已经相当详细了,可以代码结合文章细细品读。
2. 服务器启动过程
1)初始化服务器全局状态。  initServerConfig()
2)Sentinel功能初始化 initSentinel()
3)载入配置文件。    loadServerConfig()
4)创建 daemon 进程。   daemonize()
5)初始化服务器功能模块。 initServer()
6)打印 ASCII LOGO。   redisAsciiArt()
6)从AOF文件或者RDB文件中载入数据。   loadDataFromDisk()
7)开始事件循环。   aeMain()
8)服务器关闭,停止事件循环。  aeDeleteEventLoop()
其中第5部分的细节是
初始化 Redis 进程的信号功能。  setupSignalHandlers()
初始化日志功能。   openlog()
初始化客户端功能。
初始化共享对象。   createSharedObjects()
初始化事件功能。   aeCreateEventLoop()
初始化数据库。
初始化网络连接。   listenToPort()
初始化订阅与发布功能。
初始化各个统计变量。  resetServerStats()
关联服务器常规操作(cron job)到时间事件,关联客户端应答处理器到文件事件。  aeCreateTimeEvent()
如果 AOF 功能已打开,那么打开或创建 AOF 文件。
设置内存限制。
初始化cluster。    clusterInit()
初始化复制功能有关的脚本缓存。   replicationScriptCacheInit()
初始化 Lua 脚本环境。   scriptingInit()
初始化慢查询功能。    slowlogInit()
初始化后台操作线程。  bioInit()
3.命令表
一个命令用redisCommand 结构表示,redisCommandTable是所有命令的表格。

struct redisCommand redisCommandTable[] = {
{"get",getCommand,2,"r",0,NULL,1,1,1,0,0},
{"set",setCommand,-3,"wm",0,NULL,1,1,1,0,0},
//......
};
struct redisCommand {
// 命令名字
char *name;
// 实现函数
redisCommandProc *proc;
// 参数个数
int arity;
// 字符串表示的 FLAG
char *sflags; /* Flags as string representation, one char per flag. */
// 实际 FLAG
int flags;    /* The actual flags, obtained from the 'sflags' field. */
//......
};

注意redisCommandTable这种带大括号的初始化方法,
这是C语言里对struct类型的变量的一种初始化方法——顺序初始化。
顺序初始化的特点是: 按照成员定义的顺序,从前到后逐个初始化。
优点就是可以把一条记录写在一行里,不用一个个属性用.(点)的方式来一行行设置了。维护起来比较直观方便。
然后为了快速定位命令,redis用一个哈希表维护了可以执行的所有命令。
populateCommandTable函数用来初始化这张命令表,这个函数在initServerConfig里被调用。

struct redisServer {
// 命令表(受到 rename 配置选项的作用)
dict *commands;             /* Command table */
// 命令表(无 rename 配置选项的作用)
dict *orig_commands;        /* Command table before command renaming. */
};
void populateCommandTable(void) {
int j;
// 命令的数量
int numcommands = sizeof(redisCommandTable)/sizeof(struct redisCommand);
for (j = 0; j < numcommands; j++) {
// 指定命令
struct redisCommand *c = redisCommandTable+j;
int retval1, retval2;
// 将命令关联到命令表
retval1 = dictAdd(server.commands, sdsnew(c->name), c);
//将命令也关联到原始命令表
//原始命令表不会受 redis.conf 中命令改名的影响
retval2 = dictAdd(server.orig_commands, sdsnew(c->name), c);
}
}


4.命令行
redis用了一个自己写的命令行类库linenoise,只有1千行代码,但功能可谓强大,支持许多锦上添花的功能:
历史记录功能,按上下键可以查看自己输过的命令
自动补全功能,按tab键可以自动补全命令
多种颜色显示,比如输入help get帮助信息是彩色的
但是貌似没有复制粘贴功能?
其他类似的类库
名称代码行数使用它的软件
readline3万行bash,mysql,mutt
libedit2万行
linenoise1千行Redis,MongoDB,Android
jlinegroovy

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-317022-1-1.html 上篇帖子: redis的Java客户端Jedis测试验证 下篇帖子: redis支持的数据类型详解以及用途
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表