1、官网文档

https://redis.io/topics/protocol

http://www.redis.cn/topics/protocol.html

2、协议介绍

redis协议规范(Redis Protocol specification)。

redis协议在以下几点之间做出了折衷:

  (1)简单的实现

  (2)快速地被计算机解析

  (3)简单得可以能被人工解析

  (4)网络层,Redis在TCP端口6379上监听到来的连接(本质就是socket),客户端连接到来时,Redis服务器为此创建一个TCP连接。在客户端与服务器端之间传输的每个Redis命令或者数据都以\r\n结尾。

3、请求

Redis接收由不同参数组成的命令。一旦收到命令,将会立刻被处理,并回复给客户端

4、新的统一请求协议

在这个统一协议里,发送给Redis服务端的所有参数都是二进制安全的。以下是通用形式:

*后面数量表示存在几个$
$后面数量表示字符串的长度

例子:

*
$
SET
$
mykey
$
myvalue

上面的命令看上去像是单引号字符串,所以可以在查询中看到每个字节的准确值:

"*3\r\n$3\r\nSET\r\n$5\r\nmykey\r\n$7\r\nmyvalue\r\n"

在Redis的回复中也使用这样的格式。批量回复时,这种格式用于每个参数$6\r\nmydata\r\n。 实际的统一请求协议是Redis用于返回列表项,并调用 Multi-bulk回复。 仅仅是N个以以*\r\n为前缀的不同批量回复,是紧随的参数(批量回复)数目。

5、回复

Redis用不同的回复类型回复命令。它可能从服务器发送的第一个字节开始校验回复类型:

  (1)用单行回复,回复的第一个字节将是“+”

  (2)错误消息,回复的第一个字节将是“-”

  (3)整型数字,回复的第一个字节将是“:”

  (4)批量回复,回复的第一个字节将是“$”

  (5)多个批量回复,回复的第一个字节将是“*”

例子:状态回复(或者单行回复)

以“+”开始以“\r\n”结尾的单行字符串形式。例如:

"+OK\r\n"

客户端库将在“+”后面返回所有数据,正如上例中字符串“OK”一样。

有关与其他的操作请查看官方文档。

6、模拟Redis服务和客户端通讯,实现RESP协议通信

枚举类

public enum CommandRedis {
SET, GET, SETNX
}

实现类

package com.protocol;

import org.junit.Test;

import java.io.IOException;
import java.net.Proxy;
import java.net.Socket; /**
* Created by wxh on 2018/1/29.
*/
public class RedisClientByResp { private Socket socket; public RedisClientByResp() {
try {
socket = new Socket("192.168.56.180", );
} catch (IOException e) {
e.printStackTrace();
System.out.println("连接失败" + e.getMessage());
}
} /**
* 设置值
* @param key
* @param value
* @return
* @throws IOException
*/
public String set(String key, String value) throws IOException {
StringBuilder sb = new StringBuilder();
sb.append("*3").append("\r\n");
sb.append("$").append(CommandRedis.SET.name().length()).append("\r\n");
sb.append(CommandRedis.SET.name()).append("\r\n");
// 注意中文汉字。一个汉字两个字节,长度为2
sb.append("$").append(key.getBytes().length).append("\r\n");
sb.append(key).append("\r\n");
sb.append("$").append(value.getBytes().length).append("\r\n");
sb.append(value).append("\r\n");
System.out.println(sb.toString()); socket.getOutputStream().write(sb.toString().getBytes());
byte[] b = new byte[];
socket.getInputStream().read(b);
return new String(b);
} /**
* 获取值
* @param key
* @return
* @throws Exception
*/
public String get(String key) throws Exception {
StringBuffer stringBuffer = new StringBuffer();
stringBuffer.append("*2").append("\r\n");
stringBuffer.append("$").append(CommandRedis.GET.name().length()).append("\r\n");
stringBuffer.append(CommandRedis.GET).append("\r\n");
stringBuffer.append("$").append(key.getBytes().length).append("\r\n");
stringBuffer.append(key).append("\r\n"); socket.getOutputStream().write(stringBuffer.toString().getBytes());
byte[] b = new byte[];
socket.getInputStream().read(b);
return new String(b);
} /**
* 设置值:不会覆盖存在的值
* @param key
* @param value
* @return
* @throws Exception
*/
public String setnx(String key, String value) throws Exception {
StringBuffer stringBuffer = new StringBuffer();
stringBuffer.append("*3").append("\r\n");
stringBuffer.append("$").append(CommandRedis.SETNX.name().length()).append("\r\n");
stringBuffer.append(CommandRedis.SETNX.name()).append("\r\n");
stringBuffer.append("$").append(key.getBytes().length).append("\r\n");
stringBuffer.append(key).append("\r\n");
stringBuffer.append("$").append(value.getBytes().length).append("\r\n");
stringBuffer.append(value).append("\r\n"); socket.getOutputStream().write(stringBuffer.toString().getBytes());
byte[] b = new byte[];
socket.getInputStream().read(b);
return new String(b);
} public static void main(String[] args) throws Exception {
System.out.println(new RedisClientByResp().set("mykey" ,"myvalue"));
System.out.println(new RedisClientByResp().get("mykey"));
}
}

Redis随笔(六)RESP的协议规范的更多相关文章

  1. [Database] Redis 随笔

    Redis 随笔 1. 特点 非关系数据库 non-relational database 内存数据库 高性能 主从复制 可持久化存储 发布与订阅 支持脚本 2. 数据类型5种 STRING 可以是字 ...

  2. C# Redis实战(六)

    六.查询数据 在C# Redis实战(五)中介绍了如何删除Redis中数据,本篇将继续介绍Redis中查询的写法. 1.使用Linq匹配关键字查询 using (var redisClient = R ...

  3. Redis系列六:redis相关功能

    一. 慢查询原因分析 与mysql一样:当执行时间超过阀值,会将发生时间耗时的命令记录 redis命令生命周期:发送 排队 执行 返回慢查询只统计第3个执行步骤的时间 预设阀值:两种方式,默认为10毫 ...

  4. Redis学习六:Redis的持久化-AOF

    AOF(Append Only File) 一.是什么 以日志的形式来记录每个写操作,将Redis执行过的所有写指令记录下来(读操作不记录),只许追加文件但不可以改写文件,redis启动之初会读取该文 ...

  5. Redis(六)-- SpringMVC整合Redis

    一.pom.xml <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www ...

  6. 本地缓存google.guava及分布式缓存redis 随笔

    近期项目用到了缓存,我选用的是主流的google.guava作本地缓存,redis作分布式 缓存,先说说我对本地缓存和分布式缓存的理解吧,可能不太成熟的地方,大家指出,一起 学习.本地缓存的特点是速度 ...

  7. Redis(六):list/lpush/lrange/lpop 命令源码解析

    上一篇讲了hash数据类型的相关实现方法,没有茅塞顿开也至少知道redis如何搞事情的了吧. 本篇咱们继续来看redis中的数据类型的实现: list 相关操作实现. 同样,我们以使用者的角度,开始理 ...

  8. Redis系列(六)-SortedSets设计技巧

    阅读目录: 介绍 Score占位 更多位信息 总结 介绍 Redis Sorted Sets是类似Redis Sets数据结构,不允许重复项的String集合.不同的是Sorted Sets中的每个成 ...

  9. 15天玩转redis —— 第六篇 有序集合类型

    今天我们说一下Redis中最后一个数据类型 “有序集合类型”,回首之前学过的几个数据结构,不知道你会不会由衷感叹,开源的世界真好,写这 些代码的好心人真的要一生平安哈,不管我们想没想的到的东西,在这个 ...

随机推荐

  1. share pool 管理机制

    Library cache是Shared pool的一部分,它几乎是Oracle内存结构中最复杂的一部分,主要存放shared curosr(SQL)和PLSQL对象(function,procedu ...

  2. Magic Powder - 2 (CF 670_D)

    http://codeforces.com/problemset/problem/670/D2 The term of this problem is the same as the previous ...

  3. Swift3 倒计时按钮扩展

    extension UIButton{ func overrideSelf(){ self.setTitle("验证码", for: .normal) self.titleLabe ...

  4. iOS笔记之UIKit_UILable

    UILabel*label3 = [[UILabel alloc]initWithFrame:CGRectMake(0, 60+10+60+10+60+10, 320, 60)]; label3.ba ...

  5. 深拷贝&浅拷贝&引用计数&写时拷贝

    (1).浅拷贝: class String { public: String(const char* str="") :_str(]) { strcpy(_str,str); } ...

  6. unidbgrid单元格操作

    unidbgrid单元格操作 //GRID里回车替换TABfunction cellkeydown(sender, td, cellIndex, record, tr, rowIndex, e, eO ...

  7. 64位进程调用32位dll的解决方法

    64位进程调用32位dll的解决方法   最近做在Windows XP X64,VS2005环境下做32位程序编译为64位程序的工作,遇到了一些64位编程中可能遇到的问题:如内联汇编(解决方法改为C/ ...

  8. Change tab position of PageControl to bottom

    Hi, Try: UniPageControl1 -> ClientEvents -> UniEvents : function tabPanel.beforeInit(sender, c ...

  9. 在.net中使用ETW事件的方法

    直到.net4.5,才有了比较便利的操作ETW的方法. 本文介绍的方法主要来源于Microsoft.Diagnostics.Tracing.TraceEvent官方资料库. 准备 (1)需要用到类:M ...

  10. Web应用安全之Response Header里的敏感信息

    Web应用安全之Response Header 文/玄魂 目录 Web应用安全之Response Header 前言 1.1  那些敏感的header 1.2 删除敏感的header 1.2.1 删除 ...