golang实现简单的reids服务2

golang实现redis兼容的redis服务

  • 之前做的redis服务是通过tcp封装的自定义协议

原版项目地址:https://github.com/dengjiayue/my-redis.git

  • 那么能不能实现一个redis兼容的redis服务,这样一般的redis包也可以调用我们写的redis服务了呢?

当然可以,需要实现redis的RESP通信协议

新版项目地址: https://github.com/dengjiayue/my-redis-v2.0-RESP-.git

实现redis兼容的redis服务思路

  • 原本的数据处理模型不变,依旧使用单线程模型,map储存数据
  • 实现RESP协议的支持就可以了
    首先,我们需要知道redis一般收到的读写命令是什么样的去搞清楚RESP协议的原理

复制代码
"*2\r\n$3\r\nget\r\n$4\r\nname\r\n"

复制代码
"*3\r\n$3\r\nset\r\n$4\r\nname\r\n$8\r\nzhangsan\r\n"

RESP使用\r\n作为换行符

*2,*3表示命令的个数

一个命令包含前面一个命令数据的长度,比如$3 表示后面的数据长度为3; 然后在长度下一行才是数据;

一般第一个是方法名set,get什么的,第二个是key值,第三个是val值(如果是get就没有第三个),后面是过期时间什么的.

明白了工作原理我们就可以封装RESP协议支持了

  1. 根据换行符解析每一行数据
  2. 先解析第一行,获取整个请求的包含多少个命令
  3. 再解析每一个命令
  4. 先解析长度,再解析数据,
  5. 最后根据数据中的方法,key,val等消息做数据处理
  6. 封装返回:成功就返回"+{msg}\r\n",msg为处理结果;失败就返回"-Err {msg}\r\n",msg 为失败的信息

这样你就可以通过golang的redis包调用你的redis服务了

使用go-redis包做测试

go 复制代码
import (
	"context"
	"fmt"
	"time"

	"github.com/go-redis/redis/v8"
)

// 新建连接池
func NewPool() *redis.Client {
	return redis.NewClient(&redis.Options{
		Addr:         "localhost:8080",
		PoolSize:     1,
		MinIdleConns: 1,
	})
}

// 写入redis
func WriteRedis(client *redis.Client) {
	ctx := context.Background()
	// 写入redis
	rsp, err := client.Set(ctx, "name", "tom", time.Minute).Result()
	if err != nil {
		panic(err)
	}
	fmt.Println(rsp)
}

// 读取redis
func ReadRedis(client *redis.Client) {
	ctx := context.Background()
	// 读取redis
	rsp, err := client.Get(ctx, "name").Result()
	if err != nil {
		panic(err)
	}
	fmt.Println(rsp)
}

func TestWriteRedis(t *testing.T) {
	type args struct {
		client *redis.Client
	}
	tests := []struct {
		name string
		args args
	}{
		// TODO: Add test cases.
		{"test", args{NewPool()}},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			WriteRedis(tt.args.client)
		})
	}
}

func TestReadRedis(t *testing.T) {
	type args struct {
		client *redis.Client
	}
	tests := []struct {
		name string
		args args
	}{
		// TODO: Add test cases.
		{"test", args{NewPool()}},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			ReadRedis(tt.args.client)
			defer tt.args.client.Close()
		})
	}
}

// 读写测试
func TestReadWriteRedis(t *testing.T) {
	type args struct {
		client *redis.Client
	}
	tests := []struct {
		name string
		args args
	}{
		// TODO: Add test cases.
		{"test", args{NewPool()}},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			WriteRedis(tt.args.client)
			ReadRedis(tt.args.client)
			defer tt.args.client.Close()
		})
	}
}

// 读写测试
func TestReadWriteRedis(t *testing.T) {
	type args struct {
		client *redis.Client
	}
	tests := []struct {
		name string
		args args
	}{
		// TODO: Add test cases.
		{"test", args{NewPool()}},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			WriteRedis(tt.args.client)
			ReadRedis(tt.args.client)
			defer tt.args.client.Close()
		})
	}
}

读写结果

复制代码
=== RUN   TestReadWriteRedis
=== RUN   TestReadWriteRedis/test
OK
tom
--- PASS: TestReadWriteRedis/test (0.00s)
--- PASS: TestReadWriteRedis (0.00s)
PASS
ok      redis_performance_test/go_redis_read_write      0.756s

仓库地址: https://github.com/dengjiayue/my-redis-v2.0-RESP-.git

相关推荐
VBA633730 分钟前
VBA之Word应用第四章第五节:段落Paragraph对象的属性(一)
开发语言
whltaoin30 分钟前
【Java SE】Java IO体系深度剖析:从原理到实战的全方位讲解(包含流操作、序列化与 NIO 优化技巧)
java·开发语言·nio·se·io体系
csbysj20204 小时前
jQuery 删除元素
开发语言
xxy-mm4 小时前
Javascript 中的继承
开发语言·javascript·ecmascript
quikai19817 小时前
python练习第二组
开发语言·python
AI视觉网奇7 小时前
Fatal signal 11 (SIGSEGV), code 1 (SEGV_MAPERR), fault addr
开发语言·c++·算法
wjs20247 小时前
并查集快速合并
开发语言
free-elcmacom7 小时前
MATLAB与高等数学<1>一道曲面积分题的几何直观
开发语言·数学建模·matlab·高等数学
LaoZhangGong1237 小时前
深度学习uip中的“psock.c和psock.h”
c语言·开发语言
Tony Bai7 小时前
Go 安全新提案:runtime/secret 能否终结密钥残留的噩梦?
java·开发语言·jvm·安全·golang