標籤:
介紹
最近項目使用中要改造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\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
二:基礎通訊
定義配置類:
public class Configuration { public string Host { get; set; } public int Port { get; set; } /// <summary> /// Socket 是否正在使用 Nagle 演算法。 /// </summary> public bool NoDelaySocket { get; set; } public Configuration() { Host = "localhost"; Port = 6379; NoDelaySocket = false; } }
實現socket串連:
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; }
調用:
private void button1_Click(object sender, EventArgs e) { RedisBaseClient redis = new RedisBaseClient(); var result = redis.SendCommand(RedisCommand.INFO); richTextBox1.Text = result; }
響應輸出。 937是資料長度。
四: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條回複。
六:總結
本文只是簡單的實現。有興趣的同學,可以繼續下去。 ps:有點重複造輪子的感覺。
用戶端實現這塊,Socket串連池管理較複雜些。
參考資源
1:http://redis.io/topics/protocol
2:https://github.com/ServiceStack/ServiceStack.Redis
如有錯誤之處,歡迎指出糾正,對您有協助的,請推薦下 n(*≧▽≦*)n。
蘑菇先生
出處:http://www.cnblogs.com/mushroom/p/4217541.html
c#實現redis用戶端(一)