欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

c# 網(wǎng)絡編程之tcp

 更新時間:2021年02月19日 10:46:56   作者:seabluescn  
這篇文章主要介紹了c# 網(wǎng)絡編程之tcp的的相關資料,幫助大家更好的理解和學習使用c#,感興趣的朋友可以了解下

一、概述

UDP和TCP是網(wǎng)絡通訊常用的兩個傳輸協(xié)議,C#一般可以通過Socket來實現(xiàn)UDP和TCP通訊,由于.NET框架通過UdpClient、TcpListener 、TcpClient這幾個類對Socket進行了封裝,使其使用更加方便, 本文就通過這幾個封裝過的類講解一下相關應用。

二、基本應用:連接、發(fā)送、接收

服務端建立偵聽并等待連接:

TcpListener tcpListener = new TcpListener(IPAddress.Parse("127.0.0.1"), 9000);
tcpListener.Start();
if (tcpListener.Pending())
{
   TcpClient client = tcpListener.AcceptTcpClient();
   Console.WriteLine("Connected"); 
}

服務端是通過AcceptTcpClient方法獲得TcpClient對象,而客戶端是直接創(chuàng)建TcpClient對象。

TcpClient tcpClient = new TcpClient();
tcpClient.Connect("127.0.0.1", 9000);

發(fā)送數(shù)據(jù)TcpClient對象創(chuàng)建后,發(fā)送接收都通過TcpClient對象完成。

發(fā)送數(shù)據(jù):

TcpClient tcpClient = new TcpClient();
    tcpClient.Connect("127.0.0.1", 9000);
    NetworkStream netStream = tcpClient.GetStream();

    int Len = 1024;
    byte[] datas = new byte[Len];

    netStream.Write(datas, 0, Len);    

    netStream.Close();
    tcpClient.Close();

接收數(shù)據(jù):

TcpClient client = tcpListener.AcceptTcpClient();
Console.WriteLine("Connected"); 
 
 NetworkStream stream = client.GetStream();
 var remote = client.Client.RemoteEndPoint;

 byte[] data = new byte[1024];
 while (true)
 {
   if (stream.DataAvailable)
   {
    int len = stream.Read(data, 0, 1024);
    Console.WriteLine($"From:{remote}:Received ({len})");
   }
  Thread.Sleep(1);
 }

三、 粘包問題

和UDP不太一樣,TCP連接不會丟包,但存在粘包問題。(嚴格來說粘包這個說法是不嚴謹?shù)模驗門CP通訊是基于流的,沒有包的概念,包只是使用者自己的理解。) 下面分析一下粘包產(chǎn)生的原因及解決辦法。

TCP數(shù)據(jù)通訊是基于流來實現(xiàn)的,類似一個隊列,當有數(shù)據(jù)發(fā)送過來時,操作系統(tǒng)就會把發(fā)送過來的數(shù)據(jù)依次放到這個隊列中,對發(fā)送者而言,數(shù)據(jù)是一片一片發(fā)送的,所以自然會認為存在數(shù)據(jù)包的概念,但對于接收者而言,如果沒有及時去取這些數(shù)據(jù),這些數(shù)據(jù)依次存放在隊列中,彼此之間并無明顯間隔,自然就粘包了。

還有一種情況粘包是發(fā)送端造成的,有時我們調用發(fā)送代碼時,操作系統(tǒng)可能并不會立即發(fā)送,而是放到緩存區(qū),當緩存區(qū)達到一定數(shù)量時才真正發(fā)送。 

要解決粘包問題,大致有以下幾個方案。

1、 約定數(shù)據(jù)長度,發(fā)送端的數(shù)據(jù)都是指定長度,比如1024;接收端取數(shù)據(jù)時也取同樣長度,不夠長度就等待,保證取到的數(shù)據(jù)和發(fā)送端一致;

2、 接收端取數(shù)據(jù)的頻率遠大于發(fā)送端,比如發(fā)送端每1秒發(fā)送一段數(shù)據(jù),接收端每0.1秒去取一次數(shù)據(jù),這樣基本可以保證數(shù)據(jù)不會粘起來;

以上兩個方案都要求發(fā)送端需要立即發(fā)送,不可緩存數(shù)據(jù)。而且這兩種方案都有缺陷:首先,第一種方案:如果要包大小一致的話,如果約定的包比較大,肯定有較多數(shù)據(jù)冗余,浪費網(wǎng)絡資源,如果包較小,連接就比較頻繁,效率不高。

其次,第二種方案:這個方案只能在理想環(huán)境下可以實現(xiàn),當服務端遭遇一段時間的計算壓力時可能會出現(xiàn)意外,不能完全保證。

比較完善的解決方案就是對接收到的數(shù)據(jù)進行預處理:首先通過定義特殊的字符組合作為包頭和包尾,如果傳輸ASCII字符,可以用0x02表示開始(STX),用0x03表示結束(ETX),比如:STX ‘H' ‘e' ‘l' ‘l' ‘o' ETX (二進制數(shù)據(jù): 02 48 65 6C 6C 6F 03)。如果數(shù)據(jù)較長可以在包頭留出固定位置存放包長度, 如:

02 00 05 48 65 6C 6C 6F 03

其中02 05 就表示正文長度為5個字節(jié),可以進行校驗。

雖然第三種方案比較嚴謹,但相對復雜,在傳輸比較可靠、應用比較簡單的場景下,也可以采用前面兩種解決方案。

四、 一個完整的例程

服務端:

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net;
using System.Net.Sockets;
using System.Text;
using System.Threading;
using System.Threading.Tasks;

namespace TCPServer
{
 class Program
 { 
  static void Main(string[] args)
  { 
   TcpListener tcpListener = new TcpListener(IPAddress.Parse("127.0.0.1"), 9000);
   tcpListener.Start();

   while (true)
   {
    if (tcpListener.Pending())
    {
     TcpClient client = tcpListener.AcceptTcpClient();
     Console.WriteLine("Connected");     

     Task.Run(() =>
     { 
      NetworkStream stream = client.GetStream();
      var remote = client.Client.RemoteEndPoint;
      
      while (true)
      {
       if (stream.DataAvailable)
       {
        byte[] data = new byte[1024];
        int len = stream.Read(data, 0, 1024);
        string Name = Encoding.UTF8.GetString(data,0,len);
        var senddata = Encoding.UTF8.GetBytes("Hello:" + Name);
        stream.Write(senddata, 0, senddata.Length);
       }

       if (!client.IsOnline())
       {
        Console.WriteLine("Connect Closed.");
        break;
       }

       Thread.Sleep(1);
      }
     });
    }

    Thread.Sleep(1);
   }
  }
 }

 public static class TcpClientEx
 {
  public static bool IsOnline(this TcpClient client)
  {
   return !((client.Client.Poll(15000, SelectMode.SelectRead) && (client.Client.Available == 0)) || !client.Client.Connected);
  }
 }
}

客戶端:

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Linq;
using System.Net.Sockets;
using System.Text;
using System.Threading;
using System.Threading.Tasks;

namespace TCP_Clent
{
 class Program
 {
  static void Main(string[] args)
  {
   ThreadPool.SetMinThreads(100, 100);
   ThreadPool.SetMaxThreads(200, 200); 

   Parallel.For(1, 10, x =>
   {
    SendData("Tom");
   });

   Console.WriteLine("All Completed!");
   Console.ReadKey();
  }

  private static void SendData(string Name)
  {
   Task.Run(() =>
   {
    Console.WriteLine("Start");
    TcpClient tcpClient = new TcpClient();
    tcpClient.Connect("127.0.0.1", 9000);
    Console.WriteLine("Connected");
    NetworkStream netStream = tcpClient.GetStream();

    Task.Run(() =>
    {
     Thread.Sleep(100);
     while (true)
     {
      if (!tcpClient.Client.Connected)
      {
       break;
      }

      if (netStream == null)
      {
       break;
      }

      try
      {
       if (netStream.DataAvailable)
       {
        byte[] data = new byte[1024];
        int len = netStream.Read(data, 0, 1024);
        var message = Encoding.UTF8.GetString(data, 0, len);
        Console.WriteLine(message);
       }
      }
      catch
      {
       break;
      }

      Thread.Sleep(10);
     }
    });

    for (int i = 0; i < 100; i++)
    {
     byte[] datas = Encoding.UTF8.GetBytes(Name);
     int Len = datas.Length;
     netStream.Write(datas, 0, Len);
     Thread.Sleep(1000);
    }

    netStream.Close();
    netStream = null;
    tcpClient.Close();

    Console.WriteLine("Completed");
   });
  }  
 }
}

傳送門:

C#網(wǎng)絡編程入門系列包括三篇文章:

(一)C#網(wǎng)絡編程入門之UDP

(二)C#網(wǎng)絡編程入門之TCP

(三)C#網(wǎng)絡編程入門之HTTP

以上就是c# 網(wǎng)絡編程之tcp的詳細內容,更多關于c# 網(wǎng)絡編程tcp的資料請關注腳本之家其它相關文章!

相關文章

  • C#實現(xiàn)簡單的計算器功能(窗體)

    C#實現(xiàn)簡單的計算器功能(窗體)

    這篇文章主要為大家詳細介紹了C#實現(xiàn)簡單的計算器功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • C#遍歷操作系統(tǒng)下所有驅動器的方法

    C#遍歷操作系統(tǒng)下所有驅動器的方法

    這篇文章主要介紹了C#遍歷操作系統(tǒng)下所有驅動器的方法,涉及C#中DriveInfo類GetDrivers方法的使用技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-04-04
  • C#實現(xiàn)的三種模擬自動登錄和提交POST信息的方法

    C#實現(xiàn)的三種模擬自動登錄和提交POST信息的方法

    這篇文章主要介紹了C#實現(xiàn)的三種模擬自動登錄和提交POST信息的方法,分別列舉了WebBrowser、WebClient及HttpWebRequest實現(xiàn)自動登錄及提交POST的相關實現(xiàn)技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-11-11
  • C# 并行和多線程編程——認識和使用Task

    C# 并行和多線程編程——認識和使用Task

    這篇文章主要介紹了C# 并行和多線程編程——認識和使用Task的的相關資料,幫助大家更好的理解和使用c#,感興趣的朋友可以了解下
    2021-02-02
  • DataTables List互相轉換的實現(xiàn)類示例

    DataTables List互相轉換的實現(xiàn)類示例

    這篇文章主要介紹了將DataTable轉換為List,將List轉換為DataTable的實現(xiàn)類實例方法,大家參考使用吧
    2013-11-11
  • C#中委托和事件的區(qū)別詳解

    C#中委托和事件的區(qū)別詳解

    C# 中的委托(Delegate)類似于 C 或 C++ 中函數(shù)的指針。事件是在委托類型變量前加上 event 關鍵字,其本質是用來對委托類型的變量進行封裝,類似于類的屬性對字段的封裝。本文就來聊聊C#中委托和事件的區(qū)別,感興趣的可以了解一下
    2022-11-11
  • SQL Server存儲過程在C#中調用的簡單實現(xiàn)方法

    SQL Server存儲過程在C#中調用的簡單實現(xiàn)方法

    這篇文章主要給大家介紹了關于SQL Server存儲過程在C#中調用的簡單實現(xiàn)方法,文中通過示例代碼介紹的非常詳細,對大家學習或者使用SQL Server存儲過程具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧
    2020-05-05
  • C#實現(xiàn)JSON解析器MojoUnityJson功能(簡單且高效)

    C#實現(xiàn)JSON解析器MojoUnityJson功能(簡單且高效)

    MojoUnityJson 是使用C#實現(xiàn)的JSON解析器 ,算法思路來自于游戲引擎Mojoc的C語言實現(xiàn) Json.h。這篇文章主要介紹了C#實現(xiàn)JSON解析器MojoUnityJson的方法,需要的朋友可以參考下
    2018-01-01
  • C#中StringBuilder類的使用總結

    C#中StringBuilder類的使用總結

    本篇文章主要是對C#中StringBuilder類的使用方法進行了詳細的總結介紹,需要的朋友可以過來參考下,希望對大家有所幫助
    2014-01-01
  • C#用Parallel.Invoke方法盡可能并行執(zhí)行提供的每個線程

    C#用Parallel.Invoke方法盡可能并行執(zhí)行提供的每個線程

    本文主要介紹了C#用Parallel.Invoke方法盡可能并行執(zhí)行提供的每個線程,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-01-01

最新評論