使用 DotNetty 實現 Redis 的一個控制台應用程式

来源:https://www.cnblogs.com/ruxia/archive/2018/08/15/9477389.html
-Advertisement-
Play Games

零:Demo 跑出來的結果如圖 上圖說明 圖中左邊藍色的命令行界面,是用windows powershell 命令行鏈接的。 1.打開powershell命令行界面,輸入命令【telnet 127.0.0.1 6379】。 如果沒有powershell,使用cmd 命令行界面也是可以達到測試redi ...


零:Demo 跑出來的結果如圖

上圖說明

圖中左邊藍色的命令行界面,是用windows powershell 命令行鏈接的。

  1.打開powershell命令行界面,輸入命令【telnet   127.0.0.1    6379】。

   如果沒有powershell,使用cmd 命令行界面也是可以達到測試redis 命令的效果的。

   輸入PING 命令,redis 接收到,它將返回一個PONG字元串。命令的作用通常是測試與伺服器的連接是否仍然生效。PING命令

   輸入Info 命令,redis 會返回一大串的redis 服務端的信息。這個命令,主要用來測試拆包的情況,下麵會講到拆包如何處理。

圖中右邊黑色的命令行界面,是Demo 跑出來的控制台應用程式。

兩個結果一對比,測試出來,我們的Demo已經得到了正確的結果。

Ok,下麵開始進入正戲。

一 DotNetty 是什麼

DotNetty 是netty 一個C#版本。

Netty是由JBOSS提供的一個java開源框架。Netty提供非同步的、事件驅動的網路應用程式框架和工具,用以快速開發高性能、高可靠性的網路伺服器和客戶端程式。【摘自百度百科

  筆者認為 Netty是Java生態圈的一個重要組件。

  原生Socket編程,學習成本高,使用原生的Socket做項目,那就是開著一輛綠皮火車,動次打次。。。。

  使用Netty,開做項目,那開發效率無疑是高鐵般的存在。

  而且使用原生的socket 編程是很困難的

 

二,寫這個Demo 的起因

學習DotNetty很久。從DotNetty 0.4版本。到現在的0.48版本。自己實現一個C/S端的例子。還沒有太好的想法去實現。

    今天看到haifeiWu 的高作《Netty 源碼中對 Redis 協議的實現》,遂想跟著實現一個。

    所以,才有了今天的Demo.

    是的,它還只是一個Demo.並不能取代StackExchange.Redis。

三,瞭解一下redis的協議

 RESP 是 Redis 序列化協議的簡寫。它是一種直觀的文本協議,優勢在於實現非常簡單,解析性能極好。

  Redis 協議將傳輸的結構數據分為 5 種最小單元類型,單元結束時統一加上回車換行符號\r\n,來表示該單元的結束。

  單行字元串 以 + 符號開頭。

  多行字元串 以 $ 符號開頭,後跟字元串長度。

  整數值 以 : 符號開頭,後跟整數的字元串形式。

  錯誤消息 以 - 符號開頭。

  數組 以 * 號開頭,後跟數組的長度。

  關於 RESP 協議的具體介紹感興趣的小伙伴請移步 haifeiWu 的另一篇文章Redis協議規範(譯文)

  以上第二點是摘抄自 haifeiWu中的介紹

四 Demo 代碼

1,定義枚舉 RedisMessageType

 

 1 internal enum RedisMessageType:byte
 2     {
 3         /// <summary>
 4         /// 以 + 開頭的單行字元串
 5         /// </summary>
 6         SimpleString = 43,
 7 
 8         /// <summary>
 9         ///  以 - 開頭的錯誤信息
10         /// </summary>
11         Error = 45,
12         /// <summary>
13         /// 以 : 開頭的整型數據INTEGER
14         /// </summary>
15         Integer = 58,
16         /// <summary>
17         /// 以 $ 開頭的多行字元串
18         /// </summary>
19         BulkString = 36,
20 
21         /// <summary>
22         /// 以 * 開頭的數組
23         /// </summary>
24         ArrayHeader = 42
25     }
View Code

 

2,定義RedisObject   並定義了虛擬的方法 WriteBuffer

 1 public class RedisObject
 2     {
 3         public virtual void WriteBuffer(IByteBuffer output)
 4         {
 5         }
 6     }
 7 
 8 public class RedisCommon : RedisObject
 9     {
10         public RedisCommon()
11         {
12             Commond = new List<string>();
13         }
14         public List<string> Commond { get; set; }
15         public override void WriteBuffer(IByteBuffer output)
16         {
17             //請求頭部格式, *<number of arguments>\r\n
18             //const string headstr = "*{0}\r\n";
19             //參數信息       $<number of bytes of argument N>\r\n<argument data>\r\n
20             //const string bulkstr = "${0}\r\n{1}\r\n";
21             StringBuilder stringBuilder = new StringBuilder();
22             stringBuilder.AppendFormat("*{0}\r\n",Commond.Count);
23             foreach (var item in Commond)
24             {
25                 stringBuilder.AppendFormat("${0}\r\n{1}\r\n",item.Length,item);
26             }
27             //*1\r\n$4\r\nPING\r\n
28             byte[] bytes = Encoding.UTF8.GetBytes(stringBuilder.ToString());
29             output.WriteBytes(bytes);
30         }
31     }
View Code

 

3,定義RedisEncoder 編碼器, 它集成了MessageToByteEncoder<T>方法。主要是將RedisObject,寫到IByteBuffer裡面。

public class RedisEncoder:DotNetty.Codecs.MessageToByteEncoder<RedisObject>
    {
        protected override void Encode(IChannelHandlerContext context, RedisObject message, IByteBuffer output)
        {
            message.WriteBuffer(output);
            //context.WriteAndFlushAsync(output);
        }
    }

  

4,定義 RedisDecoder 解碼器,它繼承了 ByteToMessageDecoder。

  ByteToMessageDecoder 是需要自己實現解決粘包,拆包的。比較低級別,但是靈活。

  DotNetty 還有其他比較高級的解碼器。

  比如 MessageToMessageDecoder, DatagramPacketDecoder,LengthFieldBasedFrameDecoder,LineBasedFrameDecoder,ReplayingDecoder,DelimiterBasedFrameDecoder,StringDecoder。

  在李林鋒老師的《Netty權威指南》一書中,都能學習到。

  通過測試,我們知道了info 命令返回的是一個多行字元串

    以 $ 符號開頭,後跟字元串長度。假設redis 服務端要返回一個多行字元串,它的返回格式為:  ${字元串長度}\r\n{字元串}\r\n

    解析多行字元串的代碼為

  

        private string ReadMultiLine(IByteBuffer input)
        {
            Int64 strLength = ReadInteger(input);
            Int64 packLength = input.ReaderIndex + strLength + 2;
            //包的長度,比實際包還要大,跳過他,防止堆積
            if ( input.WriterIndex> packLength)
            {
                input.SkipBytes(input.ReadableBytes);
            }
            if (strLength == -1)
            {
                return null;
            }
            //包的長度,比實際包還小 拆包
            if (packLength > input.WriterIndex)
            {
                throw new Exception("");
            }
            int count = 0;
            int whildCount = 0;
            StringBuilder stringBuilder = new StringBuilder();
            while (input.IsReadable())
            {
                string str= this.ReadString(input);
                count += str.Length;
                stringBuilder.AppendLine(str);
                whildCount++;
            }

       return stringBuilder.ToString(); }

 

6.定義 RedisHandle Handler ,他繼承了SimpleChannelInboundHandler 的方法。用來接收解碼器之後解出來的RedisObJect對象。

public class RedisHandle : SimpleChannelInboundHandler<RedisObject>
    {
        protected override void ChannelRead0(IChannelHandlerContext ctx, RedisObject msg)
        {
            if (msg is ReidsString)
            {
                ReidsString reidsString = (ReidsString)msg;
                Console.WriteLine(reidsString.Content);
            }
        }
    }

 

結語:附上源碼地址

https://gitee.com/hesson/Dotnetty.Redis.Demo

感謝 @蛀牙 對本文的審閱,並提出修改的建議


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Session: 是服務端的一個鍵值對 內部機制依賴於cookie 1、分類 Django中預設支持Session,其內部提供了5種類型的Session供開發者使用: 資料庫(預設) 緩存 文件 緩存+資料庫 加密cookie 配置方式不同,操作方式相同 2、配置(settings.py) 2.1  ...
  • 1.為什麼學習Python? Python:腳本語言,易入門,可移植。 Python適用範圍:web開發、自動化測試工具編寫。 適用崗位:運維開發(運維)、自動化測試(軟體測試)、Python開發(軟體開發)。 2.Python版本和運行環境: python 2.x版本支持到2020年,現在已停止更 ...
  • uwsgi 為何還需要這東西 簡單來說,nginx屬於反向代理伺服器,他能做什麼事呢?監聽一個埠,比如說80,可以配置一個反向代理埠,比如8000,這樣,所有外部用戶對80埠的訪問實際上都是請求了8000埠的數據,只是用戶並非真實的在與8000埠交流,而是通過了80這座橋梁。目前自己只覺得 ...
  • 鎖的本質 我們先來討論鎖的出現是為瞭解決什麼問題,鎖要保證的事情其實很好理解,同一件事(一個代碼塊)在同一時刻只能由一個人(線程)操作。 這裡所說的鎖為排他鎖,暫不考慮讀寫鎖的情況 我們在這裡打個比方,假設有10個人要過獨木橋(獨木橋只能承載一個人的重量),他們可以排好隊一個一個的過,後面一個人看到 ...
  • 轉載請註明出處:http://www.cnblogs.com/zhiyong-ITNote/ 在基於dotnet core的web開發中,我們會經常用到DI,那麼如果單單使用dotnet core自身提供的一整套DI程式,該如何來實現呢?直接上代碼吧: public interface IBar { ...
  • 最近加班好累a...題外話哈 枚舉不用多說,介紹下擴展方法:擴展方法使你能夠向現有類型“添加”方法,而無需創建新的派生類型、重新編譯或以其他方式修改原始類型。 擴展方法是一種特殊的靜態方法,但可以像擴展類型上的實例方法一樣進行調用。[當然是從msdn拷貝的咯,詳情請見~] 擴展方法需定義在靜態類中, ...
  • 關鍵字用於聲明隱式的用戶定義類型轉換運算符。 如果可以確保轉換過程不會造成數據丟失,則可使用該關鍵字在用戶定義類型和其他類型之間進行隱式轉換。 引用摘自: "implicit(C 參考)" 仍以Student求和舉例 不使用 求和 使用 ...
  • 不同於隱式轉換,顯式轉換運算符必須通過轉換的方式來調用。 如果轉換操作會導致異常或丟失信息,則應將其標記為 。 這可阻止編譯器靜默調用可能產生意外後果的轉換操作。 省略轉換將導致編譯時錯誤 CS0266。 該引用摘自: "explicit(C 參考)" 顯示轉換關鍵字 能向閱讀代碼的每個人清楚地指示 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...