redis协议规范
好多年前看过redis的代码,那个时候还是2.6的版本,集群和哨兵还没加入正式代码,这几年redis发展的好快。简略翻译一篇文章redis的https://redis.io/topics/protocol
redis的客户端和服务器通过一种叫RESP (REdis Serialization Protocol)协议进行通讯。虽然他是为redis设计的协议,但是也可以用到其他的CS架构软件里。
RESP的设计主要考虑了以下几个要求:
- 容易实现
- 快速解析
- 人类可读
RESP能够序列化不同的数据类型像整型,字符串,数组。还有一种专门为错误设计的类型。从客户端发给服务器的请求是一个代表命令参数的数组。Redis返回一个该命令指定的数据类型。
RESP是二进制安全的而且从一个进程发块数据给另一个进程的时候不需要做转换,因为他会在块数据之前加上长度。
注意:这个协议只用于redis的客户端和服务器的信息交互。redis集群的node之间使用另一种二进制协议来交换信息。
网络层
redis的客户端和服务器通过6379端口建立连接。
虽然RESP是不依赖TCP协议的,但是这个协议只用在TCP连接上(或者是其他的流协议,比如unix套接字)。
请求-应答模型
redis能够接收带各种类型参数的命令。处理收到的命令并且返回应答。这是最简单的模型,但是有两个例外:
- redis支持管道,所以客户端可能一次发送多个命令然后等待应答。
- 如果redis客户端使用了发布/订阅模式(Pub/Sub),这个协议就变成了推送协议(push protocol),也就是说客户端不用再发命令了,服务器收到相应消息的时候会自动发给客户端。
除了以上两种情况,redis协议是一个简单的请求应答协议。
RESP协议描述
RESP协议是在redis的1.2版本引入的,之后在2.0版本成为标准协议,需要在客户端实现。
RESP是一种序列化协议,支持以下类型:简单字符串(sample strings),错误(Errors),整型(integers),块字符串(bulk strings)和数组(arrays)。
在RESP里,数据类型是第一个字节决定的:
- 第一个字节是“+”代表简单字符串类型。
- 第一个字节是“-”代表错误类型。
- 第一个字节是“:”代表整型。
- 第一个字节是“$”代表块字符串。
- 第一个自己是“*”代表数组。
RESP能用一个特殊的块字符串代表NULL。RESP协议不同的部分用“\r\n”(CRLF)结尾。
RESP简单字符串
简单字符串用如下方法编码:一个“+”号后面跟字符串,最后是“\r\n”,字符串里不能包含"\r\n"。简单字符串用来传输比较短的二进制安全的字符串。例如很多redis命令执行成功会返回“OK”,用RESP编码就是5个字节:
|
想要发送二进制安全的字符串,需要用RESP的块字符串。当redis返回了一个简单字符串的时候,客户端库需要给调用者返回“+”号(不含)之后CRLF之前(不含)的字符串。
RESP错误
RESP有一种专门为错误设计的类型。实际上错误类型很像RESP简单字符串类型,但是第一个字符是“-”。简单字符串类型和错误类型的区别是客户端把错误类型当成一个异常,错误类型包含的字符串是异常信息。格式是:
|
有错误发生的时候才会返回错误类型,例如你执行了一个对于某类型错误的操作,或者命令不存在等。当返回一个错误类型的时候客户端库应该发起一个异常。下面是一个错误类型的例子
|
“-”号之后空格或者换行符之前的字符串代表返回的错误类型,这只是惯例,并不是RESP要求的格式。例如ERR是一般错误,WRONGTYPE是更具体的错误表示客户端的试图在错误的类型上执行某个操作。这个称为错误前缀,能让客户端更方便的识别错误类型。
客户端可能为不同的错误返回不同的异常,也可能只提供一个一般的方法来捕捉错误并提供错误名。但是不能依赖客户端提供的这些特性,因为有的客户端仅仅返回一般错误,比如false。
RESP整型
RESP只是用一个CRLF结尾的字符串代表整型,第一个自己是“:”,例如":0\r\n"或者":1000\r\n"是整型返回。有很多redis命令返回整型,比如INCR,LLEN和LASTSVAE。返回的整型没有特定的意义,对于INCR就是一个增长后的值,对于LASTSAVE就是一个UNIX时间戳。虽然没什么意义,但是还是给他分配了一个64位的有符号的空间。整型返回值还可以用于返回true或者false,比如命令EXISTS或者SISMEMBER会返回1来代表true,返回0来代表false。其他命令比如SADD,SREM和SETNX如果执行了会返回1,没执行就返回0。如下命令也会返回整型SETNX, DEL, EXISTS, INCR, INCRBY, DECR, DECRBY, DBSIZE, LASTSAVE, RENAMENX, MOVE, LLEN, SADD, SREM, SISMEMBER, SCARD。
RESP块字符串
块字符串用来代表二进制安全的字符从,长度可达512M。块字符串用如下方式编码:
- “$”开头,跟着数字代表字符串长度,最后是CRLF。
- 字符串。
- 最后CRLF。
所以“foobar”会编码成如下方式:
|
空字符串如下:
|
RESP块字符串还能用一个特殊的格式来表示一个不存在的值,代表NULL值。在这个特殊的格式里长度是-1,不带数据。所以NULL用如下格式表示:
|
这个被称为空块字符串(NULL BULK STRING)。客户端库API不能返回空字符串,当服务端返回空块字符串的时候客户端需要返回nil。例如RUBY库需要返回‘nil’,C库需要返回NULL(或者在返回值里设置一个特殊的标志)。
RESP数组
redis客户端用RESP数组给服务器发命令。一些命令的返回值也是RESP数组类型,比如LRANGE。RESP数组使用如下格式:
- “*”号作为第一个字符,跟着一个数字代表元素个数,后面一个CRLF。
- 每个元素是一个RESP类型。
空数组如下:
|
包含两个块字符串“foo”,“bar”的数组如下:
|
*<count>CRLF在数组前面,其他的就是一个接一个的元素。例如三个整数组成的数组如下:
|
数组的元素不必是同样的类型,可以是混合的,例如包含一个包含四个整数一个块字符串的数组如下:
|
服务器第一行发送*5\r\n代表后面跟着5个元素,之后传送每一个元素。空数组是存在的,也可以用来表示空值(一般情况下用空块字符串,但是因为历史原因两种都可以用)。例如BLPOP命令超时,他会返回一个空数组,如下:
|
当服务器返回空数组以后客户端的库API应该返回一个空对象而不是空数组,区分不同状态下的空数组是有必要的。在RESP里包含数组的数组是合法的。如下,一个数组包含两个数组
|
上面是一个包含三个整数的数组和一个简单字符串数组组成的数组。
包含空元素的数组
数组里的一个元素可以为空。可以用在应答里来表明这个元素丢失了。如下数组包含一个空元素:
|
第二个元素是空,客户端库应该返回如下:
|
注意这不是上一节说的异常,而是为了进一步说明RESP协议。
给REDIS服务器发命令
现在你熟悉了RESP序列化,写一个REDIS客户端库应该不难。我们来进一步说明客户端和服务器是怎样交互的:
- 客户端给服务器端发送只包含块字符串的数组。
- 服务器会给客户端发送任何合法的RESP数据类型。
一个典型的交互是下面这样的,客户端为了获取mylist链表的长度发送LLEN mylist命令给服务器,服务器返回一个整数作为应答:
|
为了清楚我们把每一部分写在了一行,实际上客户端发送的是*2\r\n$4\r\nLLEN\r\n$6\r\nmylist\r\n。
多命令和管道
如果你需要给redis服务器发命令,但是手头只有telnet工具怎么办呢?尽管redis协议是很容易实现的,但是用这种交互型的工具实现redis协议也不是理想的工具。因此redis也可以用一种特殊的方式接受人类可读的命令,称为内联格式(inline command)。如下是客户端服务器用内联格式的例子:
|
下面是另一个内联格式,返回一个整数:
|
其实就是简单的把参数用空格分开,因为命令都不是“*”开头的,redis就会检测这种形式并且解析。
redis协议的高性能解析器
略吧,这一段没什么意思。
redis协议规范的更多相关文章
- Redis协议规范(RESP)
Redis 即 REmote Dictionary Server (远程字典服务): 而Redis的协议规范是 Redis Serialization Protocol (Redis序列化协议) 该协 ...
- Redis协议规范(译文)
Redis客户端使用名为RESP(Redis序列化协议)的协议与Redis服务器进行通信. 虽然该协议是专为Redis设计的,但它可以用于其他CS软件项目的通讯协议. RESP是以下几方面的考虑: 易 ...
- Redis随笔(六)RESP的协议规范
1.官网文档 https://redis.io/topics/protocol http://www.redis.cn/topics/protocol.html 2.协议介绍 redis协议规范(Re ...
- 从零单排学Redis【白银】
前言 只有光头才能变强 今天继续来学习Redis,上一篇从零单排学Redis[青铜]已经将Redis常用的数据结构过了一遍了.如果还没看的同学可以先去看一遍再回来~ 这篇主要讲的内容有: Redis服 ...
- 使用 DotNetty 实现 Redis 的一个控制台应用程序
零:Demo 跑出来的结果如图 上图说明 图中左边蓝色的命令行界面,是用windows powershell 命令行链接的. 1.打开powershell命令行界面,输入命令[telnet 127 ...
- Netty 源码中对 Redis 协议的实现
原文地址: haifeiWu的博客 博客地址:www.hchstudio.cn 欢迎转载,转载请注明作者及出处,谢谢! 近期一直在做网络协议相关的工作,所以博客也就与之相关的比较多,今天楼主结合 Re ...
- mini Redis(项目 二)
一个仿Redis的内存数据库(主要用来做命令解析)服务端, 客户端使用的开源工具 : https://dom4j.github.io/ github:https://github.com/h ...
- Redis【二】 set|get那些事
redis4.0.9 SET\GET方法 从哪里开始 server.c里面有每个redis命令对应的执行方法 如 struct redisCommand redisCommandTable[] = { ...
- CYQ.Data V5 分布式缓存Redis应用开发及实现算法原理介绍
前言: 自从CYQ.Data框架出了数据库读写分离.分布式缓存MemCache.自动缓存等大功能之后,就进入了频繁的细节打磨优化阶段. 从以下的更新列表就可以看出来了,3个月更新了100条次功能: 3 ...
随机推荐
- 阿里面试官:HashMap 熟悉吧?好的,那就来聊聊 Redis 字典吧!
最近,小黑哥的一个朋友出去面试,回来跟小黑哥抱怨,面试官不按套路出牌,直接打乱了他的节奏. 事情是这样的,前面面试问了几个 Java 的相关问题,我朋友回答还不错,接下来面试官就问了一句:看来 Jav ...
- Zabbix Agent升级
最近对Zabbix Server进行了升级,所以陆陆续续对Zabbix Agent也做了升级,下面是这几天工作的一个小结,鉴于经验有限和认知有限等各方面因素,下文很难面面俱到,如有疏漏或不足之处, ...
- jackson读取json tree讲解
待读取的json文本: {"data":{"count":4031,"list":[{"symbol":"SH ...
- Ubuntu 18.04 LTS 配置静态IPv6地址
学校的IPv4地址限制了校内IP访问,在家连校内机器只能先连接学校的VPN,十分不方便.好在学校没有对IPv6地址做限制,因此我们可以给自己的机器配置一个静态IPv6地址来绕过这个限制. 本机系统使用 ...
- CSS -- 元素简介
一.元素分类 在CSS中,html中的标签元素大体被分为三种不同的类型:块状元素.内联元素(又叫行内元素)和内联块状元素. 常用的块状元素有: <div>.<p>.<h1 ...
- Linux:less and Aix:more
在运维工作中,经常要查询应用日志,有Linux和Aix系统,个人感觉,Linux查询日志用less命令比较方便,Aix查询日志用more命令比较方便,在此总结一下两个命令的使用方法 AIX more命 ...
- python中的一些内置函数
1.布尔类型 2.求和sum 3.取全局变量和局部变量 4.ascii码和字符集 chr().ord() 5.看某个功能下有哪些方法 help(x).dir(x) 6.exec执行python代码 7 ...
- python3 变量
python 3变量名不能以数字开头但能数字结尾 变量名大小写敏感 在多个单词组成的变量名中以下划线间隔
- k8s滚动更新(六)
实践 滚动更新是一次只更新一小部分副本,成功后,再更新更多的副本,最终完成所有副本的更新.滚动更新的最大的好处是零停机,整个更新过程始终有副本在运行,从而保证了业务的连续性. 下面我们部署三副本应用, ...
- redis哨兵机制--配置文件sentinel.conf详解
转载自 https://blog.csdn.net/u012441222/article/details/80751390 Redis的哨兵机制是官方推荐的一种高可用(HA)方案,我们在使用Redis ...