9.通信协议

予早 2024-10-17 15:57:45
Categories: Tags:

Redis是一个CS架构的软件,通信一般分两步(不包括pipeline和PubSub):

  1. 客户端(client)向服务端(server)发送一条命令
  2. 服务端解析并执行命令,返回响应结果给客户端

客户端发送命令的格式、服务端响应结果的格式遵顼 Redis Serialization Protocol,Redis 序列化协议,简称 RESP。

RESP2

在RESP中,通过首字节的字符来区分不同数据类型

Redis支持TCP通信,可以使用Socket来模拟客户端,与Redis服务端建立连接。

import java.io.*;
import java.net.Socket;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.List;

/**
 * RESP
 */
public class Main {
    static Socket socket;
    static PrintWriter writer;
    static BufferedReader reader;

    public static void main(String[] args) {
        String host = "192.168.25.128";
        int port = 6379;
        try {
            socket = new Socket(host, port);
            writer = new PrintWriter(new OutputStreamWriter(socket.getOutputStream(), StandardCharsets.UTF_8));
            reader = new BufferedReader(new InputStreamReader(socket.getInputStream(), StandardCharsets.UTF_8));
            sendRequest("set", "name", "大哥");
            Object o = handleResponse();
            System.out.println("o = " + o);

            sendRequest("get", "name");
            o = handleResponse();
            System.out.println("o = " + o);
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (socket != null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (writer != null) {
                writer.close();
            }
            if (reader != null) {
                try {
                    reader.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }

    }

    private static Object handleResponse() throws IOException {
        int prefix = reader.read();
        switch (prefix) {
            case '+'://单行字符串
                return reader.readLine();
            case '-'://异常,一行数据
                throw new RuntimeException(reader.readLine());
            case ':'://数字
                return Long.parseLong(reader.readLine());
            case '$'://多行字符串,先读长度,再读按照长度读数据(字节流)
                int length = Integer.parseInt(reader.readLine());
                if (length == -1) return null;
                if (length == 0) return "";
            case '*':
                return readBulkString();
            default:
                throw new RuntimeException("错误的数据");
        }
    }

    private static Object readBulkString() throws IOException {
        int length = Integer.parseInt(reader.readLine());
        if ((length <= 0)) return null;
        List<Object> list = new ArrayList<>(length);
        for (int i = 0; i < length; i++) {
            list.add(handleResponse());
        }
        return list;
    }

    private static void sendRequest(String... args) {
        writer.println("*" + args.length);
        for (String arg : args) {
            writer.println("$" + arg.getBytes(StandardCharsets.UTF_8).length);
            writer.println(arg);
        }
        writer.flush();
    }
}

gossip

集群内节点通信机制

简化网络协议

刚刚提到远程缓存服务对外提供读写能力,那是对外提供的 HTTP 接口吗?
当然不是!
我们知道, HTTP 是基于 TCP 做的通信,实现了很多笨重的特性。
既然当初是为了性能,特地上的缓存服务,那就索性彻底点,抛弃 HTTP,直接基于 TCP 做传输就好!传输协议也设计得简单点,比如只要通过 TCP 传入 SET key value,就能完成写入。传入”GET key” 就能获得对应的 value。非常简洁。
那传输协议的解析需要我们自己写代码去实现吗?
完全不需要,redis 官方提供了一个命令行工具,redis-cli,通过它,我们可以输入一些命令,读写 Redis 服务器里的各种内存数据。