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

相关推荐
Evand J29 分钟前
【MATLAB例程】到达角度定位(AOA),平面环境多锚点定位(自适应基站数量),动态轨迹使用EKF滤波优化。附代码下载链接
开发语言·matlab·平面·滤波·aoa·到达角度
细节控菜鸡1 小时前
【2025最新】ArcGIS for JS 实现随着时间变化而变化的热力图
开发语言·javascript·arcgis
Pluto_CSND1 小时前
Java实现gRPC双向流通信
java·开发语言·单元测试
Yeats_Liao2 小时前
Go Web 编程快速入门 05 - 表单处理:urlencoded 与 multipart
前端·golang·iphone
原来是猿2 小时前
谈谈环境变量
java·开发语言
应用市场2 小时前
本地局域网邮件管理系统:从原理到实现的完整指南
开发语言
Tony Bai2 小时前
【Go 网络编程全解】12 本地高速公路:Unix 域套接字与网络设备信息
开发语言·网络·后端·golang·unix
oioihoii3 小时前
深入理解 C++ 现代类型推导:从 auto 到 decltype 与完美转发
java·开发语言·c++
报错小能手3 小时前
项目——基于C/S架构的预约系统平台 (1)
开发语言·c++·笔记·学习·架构
MYX_3093 小时前
第四章 多层感知机
开发语言·python