Netty 是如何解析 Redis RESP 协议的——请求篇

上文抓包分析了这个 Redis RESP 协议,看到 TCP 层面的 byte 信息,直观感受到这个 Human Readble 的二进制协议。

比如 bulk String 的格式就是: $<length>\r\n<data>\r\n

那么,剩下的特点就是 Simple to implement 和 Fast to parse。

刚好,Netty 中就有个现成的轮子,这节就一起来看看吧~

RESP2 协议

Netty 中的 Redis 模块

可以看到,整体代码并不多。

核心就在 RedisEncoder 和 RedisDecoder 上,其他就是针对这个 RESP 协议的 model。

这节是请求篇,对应 RedisEncoder ,相对比较简单,我们直接从 test 模块中找个例子来看。

RedisEncoderTest

这个代表 redis-cli ,负责发请求到这个 redis-server。

这里我摘抄了其中的 shouldEncodeSimpleString 例子,简单字符串的格式是:+OK\r\n

java 复制代码
public class RedisEncoderTest {

    private EmbeddedChannel channel;

    @BeforeEach
    public void setup() throws Exception {
        channel = new EmbeddedChannel(new RedisEncoder());
    }

    @AfterEach
    public void teardown() throws Exception {
        assertFalse(channel.finish());
    }
    
   @Test
    public void shouldEncodeSimpleString() {
        RedisMessage msg = new SimpleStringRedisMessage("simple");

        // 看这里
        boolean result = channel.writeOutbound(msg);
        assertThat(result, is(true));

        ByteBuf written = readAll(channel);
        assertThat(bytesOf(written), is(bytesOf("+simple\r\n")));
        written.release();
    }
    
    private static ByteBuf readAll(EmbeddedChannel channel) {
        ByteBuf buf = Unpooled.buffer();
        ByteBuf read;
        while ((read = channel.readOutbound()) != null) {
            buf.writeBytes(read);
            read.release();
        }
        return buf;
    }

}

图解

上面的核心流程大致如下

  1. 创建 EmbeddedChannel,并添加 RedisEncoder 这个 handler
  2. 创建 SimpleStringRedisMessage 简单字符串类型的消息
  3. 调用 writeOutbound 方法,实际会调用 handler 中的 encode 方法

这里画了个简略图 👇

下面开始源码解读 👇

创建 EmbeddedChannel

内嵌型的 Channel ,常见的还有 NioServerSockerChannel 等。

添加 RedisEncoder

new EmbeddedChannel(new RedisEncoder());

创建好这个通道后,给它添加 RedisEncoder 这个 Handler

这里的核心方法在 writeRedisMessage , 下面再聊。

创建 SimpleStringRedisMessage

java 复制代码
// 创建一个简单类型的字符串消息
// 简单字符串的格式是:`+OK\r\n` 
RedisMessage msg = new SimpleStringRedisMessage("simple");

主要提供 content 方法来获取这个字符串消息。

writeOutbound

实际去调用各个 OutboundHandler 上的 encode 方法。

这里调用到 RedisEncoder 中的 encode 方法,

writeSimpleStringMessage

还多一个 RedisMessageType.SIMPLE_STRING 参数。

RedisMessageType

一个枚举类,对应 RESP2 协议

writeString ⭐

创建 ByteBuf ,并初始化长度。

这里用 utf8 去编码这个 content,取最大值就是 3 * 6 = 18 的长度。再加上这个 + 1 个长度,EOL 两个长度,初始化时就是 21 的长度。

RedisConstants 如下

最终,写入了 9 个字节。

到了这里, channel.writeOutbound(msg); 就执行完了。

剩下就是 ByteBuf 的主场了 👇

执行 ByteBuf written = readAll(channel); 把数据读取到 ByteBuf 中,和 "+simple\r\n" 进行比较。

java 复制代码
assertThat(bytesOf(written), is(bytesOf("+simple\r\n")));

结尾

RedisEncoder 的小例子到这就结束了。

大家也可以换成下面这种,模拟常见的请求场景测试一下,上面的例子更像是服务器对客户端的应答。

java 复制代码
RedisMessage msg = new InlineCommandRedisMessage("get name");

下文再来看看这个响应篇。

本文就到这里啦,感谢您的阅读,有不对的地方也请您帮忙指正!谢谢~😋

喜欢的小伙伴们,别忘了点赞关注呀~😋 祝你有个美好的一天!😝

github.com/Java4ye 😆

相关推荐
Loo国昌1 小时前
【垂类模型数据工程】第四阶段:高性能 Embedding 实战:从双编码器架构到 InfoNCE 损失函数详解
人工智能·后端·深度学习·自然语言处理·架构·transformer·embedding
ONE_PUNCH_Ge2 小时前
Go 语言泛型
开发语言·后端·golang
良许Linux2 小时前
DSP的选型和应用
后端·stm32·单片机·程序员·嵌入式
不光头强2 小时前
spring boot项目欢迎页设置方式
java·spring boot·后端
怪兽毕设2 小时前
基于SpringBoot的选课调查系统
java·vue.js·spring boot·后端·node.js·选课调查系统
学IT的周星星2 小时前
Spring Boot Web 开发实战:第二天,从零搭个“会卖萌”的小项目
spring boot·后端·tomcat
郑州光合科技余经理2 小时前
可独立部署的Java同城O2O系统架构:技术落地
java·开发语言·前端·后端·小程序·系统架构·uni-app
Remember_9933 小时前
Spring 事务深度解析:实现方式、隔离级别与传播机制全攻略
java·开发语言·数据库·后端·spring·leetcode·oracle
好好研究4 小时前
SpringBoot整合SpringMVC
xml·java·spring boot·后端·mvc
曹轲恒4 小时前
SpringBoot整合SpringMVC(末)
java·spring boot·后端