c#实现redis客户端(一)
2017-07-14 22:17
459 查看
最近项目使用中要改造redis客户端,看了下文档,总结分享一下。
基础通信
状态命令
set、get命令
管道、事务
总结
redis允许客户端以TCP方式连接,默认6379端口。传输数据都以\r\n结尾。
例:*1\r\n$4\r\nINFO\r\n
2: 错误信息。 -开头, 例:-ERR unknown command 'mush'\r\n
3: 整型数字。 :开头, 例::1\r\n
4:大块回复值,最大512M。 $开头+数据长度。 例:$4\r\mush\r\n
5:多条回复。 *开头, 例:*2\r\n$3\r\nfoo\r\n$3\r\nbar\r\n
定义配置类:
实现socket连接:
调用:
服务端成功响应:
定义Redis命令枚举:
发送命令构建:
调用:
输出响应,其$937是数据包的长度。
调用:
输出
二者都是走的MULTI,EXEC命令,原子操作。管道就是发送命令(无需等上次命令回复),进入命令队列,然后多条命令一次执行,并返回客户端结果。
平常使用ServiceStack.Redis客户端都直接set了,其实是set、expire 2个命令。 简单实现如下:
调用:
输出:
*2 表示2条回复。
+2 表示命令执行OK。
:1 表示命令执行的结果
本文只是简单的实现,有兴趣的同学,可以继续下去。
客户端实现这块,Socket连接池管理相较复杂些。
参考资源:
http://redis.io/topics/protocol https://github.com/ServiceStack/ServiceStack.Redis
阅读目录:
协议规范基础通信
状态命令
set、get命令
管道、事务
总结
协议规范
redis允许客户端以TCP方式连接,默认6379端口。传输数据都以\r\n结尾。请求格式
*<number of arguments>\r\n$<number of bytes of argument 1>\r\n<argument data>\r\n例:*1\r\n$4\r\nINFO\r\n
响应格式
1:简单字符串,非二进制安全字符串,一般是状态回复。 +开头,例:+OK\r\n2: 错误信息。 -开头, 例:-ERR unknown command 'mush'\r\n
3: 整型数字。 :开头, 例::1\r\n
4:大块回复值,最大512M。 $开头+数据长度。 例:$4\r\mush\r\n
5:多条回复。 *开头, 例:*2\r\n$3\r\nfoo\r\n$3\r\nbar\r\n
基础通信
定义配置类:public class RedisBaseClient { //配置文件 private Configuration configuration; //通信socket private Socket socket; //接收字节数组 private byte[] ReceiveBuffer = new byte[100000]; public RedisBaseClient(Configuration config) { configuration = config; } public RedisBaseClient() : this(new Configuration()) { } public void Connect() { if (socket != null && socket.Connected) return; socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp) { NoDelay = configuration.NoDelaySocket }; socket.Connect(configuration.Host, configuration.Port); if (socket.Connected) return; Close(); } /// <summary> /// 关闭client /// </summary> public void Close() { socket.Disconnect(false); socket.Close(); } }
调用:
RedisBaseClient redis = new RedisBaseClient(); redis.Connect();
服务端成功响应:
状态命令
定义Redis命令枚举:public enum RedisCommand { GET, //获取一个key的值 INFO, //Redis信息。 SET, //添加一个值 EXPIRE, //设置过期时间 MULTI, //标记一个事务块开始 EXEC, //执行所有 MULTI 之后发的命令 }
发送命令构建:
public string SendCommand(RedisCommand command, params string[] args) { //请求头部格式, *<number of arguments>\r\n const string headstr = "*{0}\r\n"; //参数信息 $<number of bytes of argument N>\r\n<argument data>\r\n const string bulkstr = "${0}\r\n{1}\r\n"; var sb = new StringBuilder(); sb.AppendFormat(headstr, args.Length + 1); var cmd = command.ToString(); sb.AppendFormat(bulkstr, cmd.Length, cmd); foreach (var arg in args) { sb.AppendFormat(bulkstr, arg.Length, arg); } byte[] c = Encoding.UTF8.GetBytes(sb.ToString()); try { Connect(); socket.Send(c); socket.Receive(ReceiveBuffer); Close(); return ReadData(); } catch (SocketException e) { Close(); } return null; } private string ReadData() { var data = Encoding.UTF8.GetString(ReceiveBuffer); char c = data[0]; //错误消息检查。 if (c == '-') //异常处理。 throw new Exception(data); //状态回复。 if (c == '+') return data; return data; }
调用:
set、get命令
调用:private void button2_Click(object sender, EventArgs e) { RedisBaseClient redis = new RedisBaseClient(); var result = redis.SendCommand(RedisCommand.SET, "msg", "testvalue"); richTextBox1.Text = result.ToString(); } private void button3_Click(object sender, EventArgs e) { RedisBaseClient redis = new RedisBaseClient(); var result = redis.SendCommand(RedisCommand.GET, "msg"); richTextBox1.Text = result.ToString(); }
输出
管道、事务
二者都是走的MULTI,EXEC命令,原子操作。管道就是发送命令(无需等上次命令回复),进入命令队列,然后多条命令一次执行,并返回客户端结果。平常使用ServiceStack.Redis客户端都直接set了,其实是set、expire 2个命令。 简单实现如下:
public void CreatePipeline() { SendCommand(RedisCommand.MULTI, new string[] {}, true); } public string EnqueueCommand(RedisCommand command, params string[] args) { return SendCommand(command, args, true); } public string FlushPipeline() { var result = SendCommand(RedisCommand.EXEC, new string[] {}, true); Close(); return result; } public string SendCommand(RedisCommand command, string[] args, bool isPipeline=false) { //请求头部格式, *<number of arguments>\r\n const string headstr = "*{0}\r\n"; //参数信息 $<number of bytes of argument N>\r\n<argument data>\r\n const string bulkstr = "${0}\r\n{1}\r\n"; var sb = new StringBuilder(); sb.AppendFormat(headstr, args.Length + 1); var cmd = command.ToString(); sb.AppendFormat(bulkstr, cmd.Length, cmd); foreach (var arg in args) { sb.AppendFormat(bulkstr, arg.Length, arg); } byte[] c = Encoding.UTF8.GetBytes(sb.ToString()); try { Connect(); socket.Send(c); socket.Receive(ReceiveBuffer); if (!isPipeline) { Close(); } return ReadData(); } catch (SocketException e) { Close(); } return null; } public string SetByPipeline(string key, string value, int second) { this.CreatePipeline(); this.EnqueueCommand(RedisCommand.SET, key, value); this.EnqueueCommand(RedisCommand.EXPIRE, key, second.ToString()); return this.FlushPipeline(); }
调用:
private void button4_Click(object sender, EventArgs e) { RedisBaseClient redis = new RedisBaseClient(); richTextBox1.Text = redis.SetByPipeline("cnblogs", "mushroom", 1000); }
输出:
*2 表示2条回复。
+2 表示命令执行OK。
:1 表示命令执行的结果
总结
本文只是简单的实现,有兴趣的同学,可以继续下去。客户端实现这块,Socket连接池管理相较复杂些。
参考资源:
http://redis.io/topics/protocol https://github.com/ServiceStack/ServiceStack.Redis
相关文章推荐
- C# Redis客户端对一致性hash算法的实现
- 自己动手写Redis客户端(C#实现)2 - SET请求和状态回复(set)
- c#实现redis客户端(一)
- Redis系列(一):c#实现redis客户端
- 自己动手写Redis客户端(C#实现)4 - 整数回复
- c#实现redis客户端(一)
- 自己动手写Redis客户端(C#实现)3 - GET请求和批量回复
- C#实现不安装Oracle客户端访问远程oracle服务
- C#SocketAsyncEventArgs实现高效能多并发TCPSocket通信 (客户端实现)
- C#实现服务器和客户端之间通信
- .Net/C#: 实现支持断点续传多线程下载的 Http Web 客户端工具类 (C# DIY HttpWebClient)
- C# Sockets实现服务器与多个客户端进行通讯
- .Net/C#: 实现支持断点续传多线程下载的 Http Web 客户端工具类 (第2版) (C# DIY HttpWebClient)
- .Net/C# 实现 中国移动 CMPP v3.0 ISMG SP 收发短信的 SP 客户端 (第2版)(CMPP SP Client)
- 15_C# 实现VMS客户端——控件_ListView
- /* .Net/C#: 实现支持断点续传多线程下载的 Http Web 客户端工具类【转】
- C#实现不安装Oracle客户端访问远程服务器数据!!
- .Net/C#: 实现支持断点续传多线程下载的 Http Web 客户端工具类
- C# TCP实现多个客户端与服务端 数据 与 文件的传输
- C#套接字编程实现客户端和服务器通信以及传输文件