Heim Backend-Entwicklung C#.Net-Tutorial Beispiele für die Lösung von Problemen mit TCP-Sticky-Paketen in C#

Beispiele für die Lösung von Problemen mit TCP-Sticky-Paketen in C#

Jul 17, 2017 am 11:06 AM
.net 示例

Dieser Artikel stellt hauptsächlich die Lösung des TCP-Sticking-Problems in C# vor, die einen gewissen Referenzwert hat

Das Prinzip des TCP-Stickings

1. TCP-Sticky-Paket bedeutet, dass mehrere vom Sender gesendete Datenpakete in einem Paket stecken bleiben, wenn der Empfänger es empfängt. Der Header des letzteren Datenpakets folgt sofort Ende des vorherigen Datenpakets. Es gibt viele Gründe für das Phänomen der Sticky Packets. Es kann sowohl beim Absender als auch beim Empfänger liegen.

2. Das vom Absender verursachte Sticky Packet wird durch das TCP-Protokoll selbst verursacht. Um die Übertragungseffizienz von TCP zu verbessern, muss der Absender häufig genügend Daten sammeln, bevor er ein Datenpaket sendet. Wenn die mehrmals hintereinander gesendeten Daten sehr klein sind, kombiniert TCP die Daten normalerweise gemäß dem Optimierungsalgorithmus zu einem Paket und sendet es sofort aus, sodass der Empfänger die Sticky-Paketdaten empfängt. Das vom Empfänger verursachte Sticky-Paket ist darauf zurückzuführen, dass der Benutzerprozess des Empfängers die Daten nicht rechtzeitig empfängt, was zum Phänomen des Sticky-Pakets führt.

3. Dies liegt daran, dass der Empfänger die empfangenen Daten zuerst in den Empfangspuffer des Systems legt und der Benutzerprozess die Daten aus dem Puffer entnimmt. Wenn das nächste Datenpaket eintrifft, ist dies beim vorherigen Datenpaket nicht der Fall Wenn es vom Benutzerprozess empfangen wurde, wird das nächste Datenpaket nach dem Empfang des vorherigen Datenpakets im Systemempfangspuffer abgelegt, und der Benutzerprozess ruft Daten entsprechend der voreingestellten Puffergröße aus dem Systemempfangspuffer ab. so dass mehrere Datenpakete gleichzeitig abgerufen werden. ,

2. Lösungsprinzip und Code-Implementierung

1. Verwenden Sie den Header (feste Länge, die die Länge des Paketkörpers enthält, die beim Senden dynamisch erhalten wird) + Übertragungsmechanismus des Paketkörpers. Wie in der Abbildung gezeigt

HeaderSize speichert die Länge des Paketkörpers und die HeaderSize selbst hat eine feste Länge von 4 Bytes

Eine vollständige Datenmenge packet (L) = HeaderSize+BodySize;

2. Subpackaging-Algorithmus

Die Grundidee besteht darin, zunächst den zu verarbeitenden empfangenen Datenstrom, also die Systempufferdaten (Länge), zwangsweise umzuwandeln auf M gesetzt) ​​in eine vorgegebene Strukturdatenform umwandeln, das Strukturdatenlängenfeld L daraus extrahieren und dann die erste Paketdatenlänge basierend auf dem Paketheader berechnen.

M=Systempuffergröße; L=vom Benutzer gesendetes Datenpaket=HeaderSize+BodySize;

1) Wenn L

2) Wenn L=M, bedeutet dies, dass der Inhalt des Datenstroms zufällig eine vollständige Datenstruktur ist (d. h. der benutzerdefinierte Puffer ist gleich). an die System-Empfangspuffergröße), direkt Speichern Sie es einfach im temporären Puffer.

3) Wenn L>M, bedeutet dies, dass der Inhalt des Datenstroms nicht ausreicht, um vollständige strukturierte Daten zu bilden, und mit dem nächsten Paket zusammengeführt werden muss Daten vor der Verarbeitung.

4) Das Folgende ist die Code-Implementierung (die Serverseite des HP-SOCKET-Frameworks zum Empfangen von Daten)

int headSize = 4;//包头长度 固定4
  byte[] surplusBuffer = null;//不完整的数据包,即用户自定义缓冲区
  /// <summary>
  /// 接收客户端发来的数据
  /// </summary>
  /// <param name="connId">每个客户的会话ID</param>
  /// <param name="bytes">缓冲区数据</param>
  /// <returns></returns>
  private HandleResult OnReceive(IntPtr connId, byte[] bytes) 
  {
   //bytes 为系统缓冲区数据
   //bytesRead为系统缓冲区长度
   int bytesRead = bytes.Length;
   if (bytesRead > 0)
   {
    if (surplusBuffer == null)//判断是不是第一次接收,为空说是第一次
     surplusBuffer = bytes;//把系统缓冲区数据放在自定义缓冲区里面
    else
     surplusBuffer = surplusBuffer.Concat(bytes).ToArray();//拼接上一次剩余的包
    //已经完成读取每个数据包长度
    int haveRead = 0;
    //这里totalLen的长度有可能大于缓冲区大小的(因为 这里的surplusBuffer 是系统缓冲区+不完整的数据包)
    int totalLen = surplusBuffer.Length;
    while (haveRead <= totalLen)
    {
     //如果在N此拆解后剩余的数据包连一个包头的长度都不够
     //说明是上次读取N个完整数据包后,剩下的最后一个非完整的数据包
     if (totalLen - haveRead < headSize)
     {
      byte[] byteSub = new byte[totalLen - haveRead];
      //把剩下不够一个完整的数据包存起来
      Buffer.BlockCopy(surplusBuffer, haveRead, byteSub, 0, totalLen - haveRead);
      surplusBuffer = byteSub;
      totalLen = 0;
      break;
     }
     //如果够了一个完整包,则读取包头的数据
     byte[] headByte = new byte[headSize];
     Buffer.BlockCopy(surplusBuffer, haveRead, headByte, 0, headSize);//从缓冲区里读取包头的字节
     int bodySize = BitConverter.ToInt32(headByte, 0);//从包头里面分析出包体的长度

     //这里的 haveRead=等于N个数据包的长度 从0开始;0,1,2,3....N
     //如果自定义缓冲区拆解N个包后的长度 大于 总长度,说最后一段数据不够一个完整的包了,拆出来保存
     if (haveRead + headSize + bodySize > totalLen)
     {
      byte[] byteSub = new byte[totalLen - haveRead];
      Buffer.BlockCopy(surplusBuffer, haveRead, byteSub, 0, totalLen - haveRead);
      surplusBuffer = byteSub;
      break;
     }
     else
     {
      //挨个分解每个包,解析成实际文字
      String strc = Encoding.UTF8.GetString(surplusBuffer, haveRead + headSize, bodySize);
      //AddMsg(string.Format(" > [OnReceive] -> {0}", strc));
      //依次累加当前的数据包的长度
      haveRead = haveRead + headSize + bodySize;
      if (headSize + bodySize == bytesRead)//如果当前接收的数据包长度正好等于缓冲区长度,则待拼接的不规则数据长度归0
      {
       surplusBuffer = null;//设置空 回到原始状态
       totalLen = 0;//清0
      }
     }
    }
   }
   return HandleResult.Ok;
  }
Nach dem Login kopieren

Zu diesem Zeitpunkt Das Entpacken und Parsen des Textes ist abgeschlossen. Aber es ist noch nicht wirklich fertig. Wenn dieser Code dafür vorgesehen ist, dass der Client Daten vom Server empfängt, ist das in Ordnung.

Sehen Sie sich die IntPtr-ConnId-Sitzungs-ID jeder Verbindung sorgfältig an

private HandleResult OnReceive(IntPtr connId, byte[] bytes)
{
}
Nach dem Login kopieren

Aber die Serverseite muss auch unterscheiden, aus welcher Sitzung jedes Datenpaket generiert wird, da die Serverseite mehrere Threaded und Multithreaded Im Benutzermodus können das erste und das zweite Paket aus unterschiedlichen Sitzungen stammen, sodass der obige Code nur im Einzelsitzungsmodus funktioniert.

Ich werde dieses Problem jetzt lösen.

Verwenden von c#sicherConaktuellWörterbuch,

neuester Code

//线程安全的字典
  ConcurrentDictionary<IntPtr, byte[]> dic = new ConcurrentDictionary<IntPtr, byte[]>();
  int headSize = 4;//包头长度 固定4
  /// <summary>
  /// 接收客户端发来的数据
  /// </summary>
  /// <param name="connId">每个客户的会话ID</param>
  /// <param name="bytes">缓冲区数据</param>
  /// <returns></returns>
  private HandleResult OnReceive(IntPtr connId, byte[] bytes) 
  {
   //bytes 为系统缓冲区数据
   //bytesRead为系统缓冲区长度
   int bytesRead = bytes.Length;
   if (bytesRead > 0)
   {
    byte[] surplusBuffer = null;
    if (dic.TryGetValue(connId, out surplusBuffer))
    {
     byte[] curBuffer = surplusBuffer.Concat(bytes).ToArray();//拼接上一次剩余的包
     //更新会话ID 的最新字节
     dic.TryUpdate(connId, curBuffer, surplusBuffer);
     surplusBuffer = curBuffer;//同步
    }
    else
    {
     //添加会话ID的bytes
     dic.TryAdd(connId, bytes);
     surplusBuffer = bytes;//同步
    }

    //已经完成读取每个数据包长度
    int haveRead = 0;
    //这里totalLen的长度有可能大于缓冲区大小的(因为 这里的surplusBuffer 是系统缓冲区+不完整的数据包)
    int totalLen = surplusBuffer.Length;
    while (haveRead <= totalLen)
    {
     //如果在N此拆解后剩余的数据包连一个包头的长度都不够
     //说明是上次读取N个完整数据包后,剩下的最后一个非完整的数据包
     if (totalLen - haveRead < headSize)
     {
      byte[] byteSub = new byte[totalLen - haveRead];
      //把剩下不够一个完整的数据包存起来
      Buffer.BlockCopy(surplusBuffer, haveRead, byteSub, 0, totalLen - haveRead);
      dic.TryUpdate(connId, byteSub, surplusBuffer);
      surplusBuffer = byteSub;
      totalLen = 0;
      break;
     }
     //如果够了一个完整包,则读取包头的数据
     byte[] headByte = new byte[headSize];
     Buffer.BlockCopy(surplusBuffer, haveRead, headByte, 0, headSize);//从缓冲区里读取包头的字节
     int bodySize = BitConverter.ToInt32(headByte, 0);//从包头里面分析出包体的长度

     //这里的 haveRead=等于N个数据包的长度 从0开始;0,1,2,3....N
     //如果自定义缓冲区拆解N个包后的长度 大于 总长度,说最后一段数据不够一个完整的包了,拆出来保存
     if (haveRead + headSize + bodySize > totalLen)
     {
      byte[] byteSub = new byte[totalLen - haveRead];
      Buffer.BlockCopy(surplusBuffer, haveRead, byteSub, 0, totalLen - haveRead);
      dic.TryUpdate(connId, byteSub, surplusBuffer);
      surplusBuffer = byteSub;
      break;
     }
     else
     {
      //挨个分解每个包,解析成实际文字
      String strc = Encoding.UTF8.GetString(surplusBuffer, haveRead + headSize, bodySize);
      AddMsg(string.Format(" > {0}[OnReceive] -> {1}", connId, strc));
      //依次累加当前的数据包的长度
      haveRead = haveRead + headSize + bodySize;
      if (headSize + bodySize == bytesRead)//如果当前接收的数据包长度正好等于缓冲区长度,则待拼接的不规则数据长度归0
      {
       byte[] xbtye=null;
       dic.TryRemove(connId, out xbtye);
       surplusBuffer = null;//设置空 回到原始状态
       totalLen = 0;//清0
      }
     }
    }
   }
   return HandleResult.Ok;
  }
Nach dem Login kopieren

Dies löst das Problem, viel Empfang Verwirrung durch Client-Sitzung. Zu diesem Zeitpunkt sind alle Arbeiten abgeschlossen. Der obige Code dient nur als Referenz und zum Lernen, wenn Sie sich wirklich nicht solche Mühe machen wollen. Sie können direkt das PACK-Modell des HP-SOCKET-Kommunikationsframeworks verwenden, das das Problem der Sticky-Pakete automatisch löst.

Das obige ist der detaillierte Inhalt vonBeispiele für die Lösung von Problemen mit TCP-Sticky-Paketen in C#. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn

Heiße KI -Werkzeuge

Undresser.AI Undress

Undresser.AI Undress

KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover

AI Clothes Remover

Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool

Undress AI Tool

Ausziehbilder kostenlos

Clothoff.io

Clothoff.io

KI-Kleiderentferner

AI Hentai Generator

AI Hentai Generator

Erstellen Sie kostenlos Ai Hentai.

Heißer Artikel

R.E.P.O. Energiekristalle erklärten und was sie tun (gelber Kristall)
3 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Beste grafische Einstellungen
3 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. So reparieren Sie Audio, wenn Sie niemanden hören können
3 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌

Heiße Werkzeuge

Notepad++7.3.1

Notepad++7.3.1

Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version

SublimeText3 chinesische Version

Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1

Senden Sie Studio 13.0.1

Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6

Dreamweaver CS6

Visuelle Webentwicklungstools

SublimeText3 Mac-Version

SublimeText3 Mac-Version

Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

Einführung in Python-Funktionen: Einführung und Beispiele der Exec-Funktion Einführung in Python-Funktionen: Einführung und Beispiele der Exec-Funktion Nov 03, 2023 pm 02:09 PM

Einführung in Python-Funktionen: Einführung und Beispiele der Exec-Funktion Einführung: In Python ist Exec eine integrierte Funktion, die zum Ausführen von Python-Code verwendet wird, der in einer Zeichenfolge oder Datei gespeichert ist. Die exec-Funktion bietet eine Möglichkeit, Code dynamisch auszuführen, sodass das Programm während der Laufzeit nach Bedarf Code generieren, ändern und ausführen kann. In diesem Artikel wird die Verwendung der Exec-Funktion vorgestellt und einige praktische Codebeispiele gegeben. So verwenden Sie die Exec-Funktion: Die grundlegende Syntax der Exec-Funktion lautet wie folgt: exec

Spezifikationen und Beispiele für Go-Spracheinrückungen Spezifikationen und Beispiele für Go-Spracheinrückungen Mar 22, 2024 pm 09:33 PM

Einrückungsspezifikationen und Beispiele der Go-Sprache Die von Google entwickelte Programmiersprache ist für ihre prägnante und klare Syntax bekannt, bei der Einrückungsspezifikationen eine entscheidende Rolle für die Lesbarkeit und Schönheit des Codes spielen. In diesem Artikel werden die Einrückungsspezifikationen der Go-Sprache vorgestellt und anhand spezifischer Codebeispiele ausführlich erläutert. Einrückungsspezifikationen In der Go-Sprache werden Tabulatoren anstelle von Leerzeichen zum Einrücken verwendet. Jede Einrückungsebene besteht aus einem Tabulator, der normalerweise auf eine Breite von 4 Leerzeichen eingestellt ist. Solche Spezifikationen vereinheitlichen den Codierungsstil und ermöglichen es Teams, beim Kompilieren zusammenzuarbeiten

Ausführliche Erklärung und Anwendungsbeispiele der Oracle DECODE-Funktion Ausführliche Erklärung und Anwendungsbeispiele der Oracle DECODE-Funktion Mar 08, 2024 pm 03:51 PM

Die DECODE-Funktion in Oracle ist ein bedingter Ausdruck, der häufig verwendet wird, um unterschiedliche Ergebnisse basierend auf unterschiedlichen Bedingungen in Abfrageanweisungen zurückzugeben. In diesem Artikel werden die Syntax, Verwendung und der Beispielcode der DECODE-Funktion im Detail vorgestellt. 1. Syntax der DECODE-Funktion DECODE(expr,search1,result1[,search2,result2,...,default]) expr: der zu vergleichende Ausdruck oder das zu vergleichende Feld. Suche1,

Einführung in Python-Funktionen: Verwendung und Beispiele der abs-Funktion Einführung in Python-Funktionen: Verwendung und Beispiele der abs-Funktion Nov 03, 2023 pm 12:05 PM

Einführung in Python-Funktionen: Verwendung und Beispiele der abs-Funktion 1. Einführung in die Verwendung der abs-Funktion In Python ist die abs-Funktion eine integrierte Funktion, die zur Berechnung des Absolutwerts eines bestimmten Werts verwendet wird. Es kann ein numerisches Argument akzeptieren und den absoluten Wert dieser Zahl zurückgeben. Die grundlegende Syntax der abs-Funktion lautet wie folgt: abs(x) wobei x der numerische Parameter zur Berechnung des Absolutwerts ist, der eine Ganzzahl oder eine Gleitkommazahl sein kann. 2. Beispiele für die abs-Funktion Im Folgenden zeigen wir die Verwendung der abs-Funktion anhand einiger spezifischer Beispiele: Beispiel 1: Berechnung

Teilen Sie mehrere .NET-Open-Source-KI- und LLM-bezogene Projekt-Frameworks Teilen Sie mehrere .NET-Open-Source-KI- und LLM-bezogene Projekt-Frameworks May 06, 2024 pm 04:43 PM

Die Entwicklung von Technologien der künstlichen Intelligenz (KI) ist heute in vollem Gange und sie haben in verschiedenen Bereichen großes Potenzial und Einfluss gezeigt. Heute wird Dayao Ihnen 4 .NET Open-Source-KI-Modell-LLM-bezogene Projekt-Frameworks vorstellen und hofft, Ihnen einige Referenzen zu geben. https://github.com/YSGStudyHards/DotNetGuide/blob/main/docs/DotNet/DotNetProjectPicks.mdSemanticKernelSemanticKernel ist ein Open-Source-Softwareentwicklungskit (SDK), das für die Integration großer Sprachmodelle (LLM) wie OpenAI und Azure entwickelt wurde

Einführung in Python-Funktionen: Funktionen und Beispiele für Auswertungsfunktionen Einführung in Python-Funktionen: Funktionen und Beispiele für Auswertungsfunktionen Nov 04, 2023 pm 12:24 PM

Einführung in Python-Funktionen: Funktionen und Beispiele der Eval-Funktion In der Python-Programmierung ist die Eval-Funktion eine sehr nützliche Funktion. Die Eval-Funktion kann eine Zeichenfolge als Programmcode ausführen und ihre Funktion ist sehr leistungsfähig. In diesem Artikel stellen wir die detaillierten Funktionen der Bewertungsfunktion sowie einige Anwendungsbeispiele vor. 1. Funktion der Eval-Funktion Die Funktion der Eval-Funktion ist sehr einfach. Sie kann einen String als Python-Code ausführen. Das bedeutet, dass wir einen String konvertieren können

Wie sind die Berufsaussichten von C#? Wie sind die Berufsaussichten von C#? Oct 19, 2023 am 11:02 AM

Ganz gleich, ob Sie Anfänger oder erfahrener Profi sind: Die Beherrschung von C# ebnet den Weg für Ihre Karriere.

Einführung in Python-Funktionen: Verwendung und Beispiele der isinstance-Funktion Einführung in Python-Funktionen: Verwendung und Beispiele der isinstance-Funktion Nov 04, 2023 pm 03:15 PM

Einführung in Python-Funktionen: Verwendung und Beispiele der isinstance-Funktion Python ist eine leistungsstarke Programmiersprache, die viele integrierte Funktionen bereitstellt, um das Programmieren komfortabler und effizienter zu gestalten. Eine der sehr nützlichen integrierten Funktionen ist die Funktion isinstance(). In diesem Artikel werden die Verwendung und Beispiele der Funktion isinstance vorgestellt und spezifische Codebeispiele bereitgestellt. Mit der Funktion isinstance() wird ermittelt, ob ein Objekt eine Instanz einer bestimmten Klasse oder eines bestimmten Typs ist. Die Syntax dieser Funktion ist wie folgt

See all articles