4.1、客户端通信协议

Redis是用单线程来处理多个客户端的访问,因此作为Redis的开发和运维人员需要了解Redis服务端和客户
端的通信协议,以及主流编程语言的Redis客户端使用方法,同时还需要了解客户端管理的相应API以及开发
运维中可能遇到的问题。本章内容如下:

  • 客户端通信协议
  • Java客户端Jedis
  • Python客户端redis-py
  • 客户端管理
  • 客户端常见异常
  • 客户端案例分析

客户端通信协议

几乎所有的主流编程语言都有Redis的客户端,不考虑Redis非常流行的原因,如果站在技术的角度看原因还
有两个:第一、客户端与服务端之间的通信协议是在TCP协议之上构建的。第二,Redis制定了RESP
(Redis Serialization Protocol,Redis序列化协议)实现客户端与服务端的正常交互,这种协议简单
高效,既能够被机器解析,又容易被人类识别。例如客户端发送一条set hello world命令给服务端,按照
RESP的标准,客户端需要将其封装为如下格式(每行用\r\n间隔):

*3
$3
SET
$5
hello
$5
world

这样Redis服务端能够按照RESP将其解析为set hello world命令,执行后回复的格式如下:

+OK

可以看到除了命令(set hello world)和返回结果(OK)本身还包含了一些特殊字符以及数字,下面将对
这些格式进行说明。

  1. 发送命令格式

    RESP的规定一条命令的格式如下,CRLF代表"\r\n"。

    *<参数数量> CRLF
    $<参数1的字节数量> CRLF
    <参数1> CRLF
    ...
    $<参数N的字节数量> CRLF
    <参数N> CRLF
    

    依然以set hello world这条命令进行说明。
    参数数量为3个,因此第一行为:

    *3
    

    参数字节数分别是3 5 5,因此后面几行为:

    $3
    SET
    $5
    hello
    $5
    world
    

    有一点要注意的是,上面只是格式化显示的结果,实际传输格式为如下代码,整个过程如下图所示:

    *3\r\n$3\r\nSET\r\n$5\r\nhello\r\n$5\r\nworld\r\n
    
    2019-04-11-21-32-22.png
  2. 返回结果格式

    Redis的返回结果类型分为以下五种,如下图所示:

    • 状态恢复:在RESP中第一个字节为“+”。
    • 错误回复:在RESP中第一个字节为“-”。
    • 整数回复:在RESP中第一个字节为“:”。
    • 字符串回复:在RESP中第一个字节为“$”。
    • 多条字符串回复:在RESP中第一个字节为“*”。
    2019-04-11-21-45-40.png

    我们知道redis-cli只能看到最终的执行结果,那是因为redis-cli本身就是按照RESP进行结果解析
    的,所以看不到中间结果,redis-cli.c源码命令结果的解析结构如下:

    static sds cliFormatReplyTTY(redisReply *r, char *prefix){
        sds out = sdsempty();
        switch (r->type){
        case REDIS_REPLY_ERROR:
            //处理错误回复
        case REDIS_REPLY_STATUS:
            //处理状态回复
        case REDIS_REPLY_INTEGER:
            //处理整数回复
        case REDIS_REPLY_STRING:
            //处理字符串回复
        case REDIS_REPLY_NIL:
            //处理空
        case REDIS_REPLY_ARRAY:
            //处理多条字符串回复
        return out;
        }
    }
    

    例如执行set hello world,返回结果是OK,并不能看到加号:

    127.0.0.1:6379> set hello world
    OK
    

    为了看到Redis服务端返回的“真正”结果,可以使用nc命令、telnet命令、甚至写一个socket程序进行
    模拟。下面以nc命令进行演示,首先使用nc 127.0.0.1:6379连接到Redis:

    nc 127.0.0.1:6379

    状态回复:set hello world的返回结果为+OK:

    set hello world
    +OK
    

    错误回复:由于sethx这条命令不存在,那么返回结果就是"-"号加上错误消息:

    sethx
    -ERR unknown command 'sethx'
    

    整数回复:当命令的执行结果是整数时,返回结果就是整数回复,例如incr、exsists、del、dbsize
    返回结果是整数,例如执行incr counter返回结果就是":"加上整数:

    incr counter
    :1
    

    字符串回复:当命令的执行结果是字符串是,返回结果就是字符串回复。例如get、hget返回结果都是
    字符串,例如get hello的结果为"$5\r\nworld\r\n":

    get hello
    $5
    world
    

    多条字符串回复:当命令的执行结果是多条字符串是,返回结果就是多条字符串回复。
    例如mget、hgetall、lrange等命令会返回多个结果,例如下面操作:
    首先使用mset设置多个键值对:

    mset java jedis python redis-py
    +OK
    

    然后执行mget命令返回多个结果,第一个*2代表返回结果的个数,后面的格式是和字符串回复一致的:

    mget java python
    *2
    $5
    jedis
    $8
    redis-py
    

    有一点需要注意,无论是字符串回复还是多条字符串回复,如果有nil值,那么会返回$-1。

    例如,对一个不存在的键执行get操作,返回结果为:

    get not_exist_key
    $-1
    

    如果批量操作中包含一条为nil值的结果,那么返回结果如下:

    mget hello not_exists_key java
    *3
    $5
    world
    $-1
    $5
    jedis
    

    有了RESP提供的发送命令和返回结果的协议格式,各种编程语言就可以利用其来实现相应的Redis客服

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 224,242评论 6 522
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 95,953评论 3 402
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 171,299评论 0 366
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 60,709评论 1 300
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 69,723评论 6 399
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 53,236评论 1 314
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 41,629评论 3 428
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 40,594评论 0 279
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 47,135评论 1 324
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 39,156评论 3 345
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 41,285评论 1 354
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,914评论 5 350
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,600评论 3 336
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 33,073评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 34,203评论 1 275
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 49,798评论 3 381
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 46,339评论 2 365

推荐阅读更多精彩内容