使用 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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...