TCP解决粘包分包,简单封装

1.TCP粘包拆包原因

1.1TCP粘包原因

TCP粘包是指在数据传输过程中,短时间发送多个小数据包被合并成一个大的数据包,

包1
包2包3
包4

1.2TCP分包原因

TCP分包发送方一次发送大数据包被拆分成多个小的数据包,

包1_1
包1_2

1.3解决方法

1.包固定长度
2.包结尾特定字符分割如\n
3.消息头包含包长度

2.具体实现

在这使用特殊字符分割,缓存数组,
1.在拿到多个分隔符时候,通过分隔符解析多条消息。
2.在没有拿到分隔符时候,继续缓存,直到拿到分隔符,保证完整包。

服务器代码:

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

class Server
{
    static string delimiter = "\r\n";
    static List<byte> leftoverData = new List<byte>();

    public void Start()
    {
        int port = 1234;
        TcpListener listener = new TcpListener(IPAddress.Any, port);
        listener.Start();
        Console.WriteLine("Server started. Waiting for connections...");

        TcpClient client = listener.AcceptTcpClient();
        Console.WriteLine("Client connected.");

        // 启动发送消息的任务
        Task sendTask = Task.Run(() => SendMessages(client));

        // 启动接收消息的任务
        Task receiveTask = Task.Run(() => ReceiveMessages(client));

        // 等待发送和接收任务完成
        Task.WaitAll(sendTask, receiveTask);

        client.Close();
        listener.Stop();
        Console.WriteLine("Server stopped.");
    }

    static void SendMessages(TcpClient client)
    {
        NetworkStream stream = client.GetStream();
        string[] messages = { "Hello", "以下是移除了文本中所有换行和空格的版本:当我们展望未来,科技的进步和创新无疑将引领着人类走向一个全新的世界。以下是一段关于未来科技的随机文本,长约800字:未来科技的世界将是一个充满奇迹和无限可能的地方。人工智能、生物技术、量子计算等前沿领域的突破将改变我们的生活方式、推动社会发展,并带来前所未有的机遇和挑战。人工智能将成为未来的核心技术。智能机器人将在各个领域发挥重要作用,从医疗保健到交通运输,从教育到家庭生活,无所不在。智能机器人不仅能够执行各种任务,还能与人类进行交互和合作。他们将成为我们的助手、朋友和教师,为我们提供个性化的服务和支持。生物技术的进步将彻底改变医学和生命科学。基因编辑技术的突破将使得人类能够更精确地治疗疾病,并可能解决一些遗传性疾病的根本问题。仿生技术的发展将使得人类能够模仿自然界的优秀设计,创造出更强大、更灵活的机器和材料。生物打印技术将使得人体器官的定制化制造成为可能,解决器官移植的瓶颈问题。量子计算的崛起将彻底改变计算机科学和信息技术。量子计算机的运算速度将远远超过传统计算机,解决那些传统计算机无法应对的复杂问题。密码学、材料科学、化学合成等领域将从量子计算的进步中获益。同时,量子通信技术将实现绝对安全的通信,保护个人隐私和商业机密。能源技术的创新将推动可持续发展和应对气候变化。太阳能、风能等可再生能源将成为主流,取代传统的化石燃料。能源储存技术的突破将解决可再生能源的不稳定性问题,实现可持续能源的大规模应用。核聚变技术的实现将为人类提供清洁、高效、安全的能源来源。虚拟现实和增强现实技术将改变人们的感知和体验。我们将进入一个全新的数字世界,与虚拟对象进行互动、探索虚拟环境、参与虚拟社交。", "12345" };

        foreach (string message in messages)
        {
            byte[] data = Encoding.UTF8.GetBytes(message + delimiter);
            stream.Write(data, 0, data.Length);
            Console.WriteLine("Sent message: " + message);

            Task.Delay(1000).Wait(); // 模拟发送间隔
        }

        // for (int i = 0; i < 5000; i++)
        // {
        //     byte[] data = Encoding.UTF8.GetBytes($"server back:{i}{delimiter}" );
        //     stream.Write(data, 0, data.Length);
        // }
    }

    static async Task ReceiveMessages(TcpClient client)
    {
        NetworkStream stream = client.GetStream();
        byte[] buffer = new byte[4096];
        int bytesRead = 0;

        while (true)
        {
            try
            {
                bytesRead = await stream.ReadAsync(buffer, 0, buffer.Length);
                if (bytesRead == 0)
                {
                    break;
                }

                ProcessReceivedData(buffer, bytesRead);
            }
            catch (Exception ex)
            {
                Console.WriteLine("Error reading data: " + ex.Message);
                break;
            }
        }
    }

    static void ProcessReceivedData(byte[] receivedBytes, int bytesRead)
    {
        leftoverData.AddRange(receivedBytes.Take(bytesRead));

        while (true)
        {
            int delimiterIndex = FindDelimiterIndex(leftoverData, delimiter);

            if (delimiterIndex == -1)
            {
                break; // 没有找到完整的消息
            }

            byte[] messageBytes = leftoverData.GetRange(0, delimiterIndex).ToArray();
            string message = Encoding.UTF8.GetString(messageBytes);
            Console.WriteLine("Received message: " + message);

            // 在这里进行消息处理逻辑
            // ...

            leftoverData.RemoveRange(0, delimiterIndex + delimiter.Length);
        }
    }

    static int FindDelimiterIndex(List<byte> data)
    {
        byte[] delimiterBytes = Encoding.UTF8.GetBytes(delimiter);
        int delimiterLength = delimiterBytes.Length;

        for (int i = 0; i <= data.Count - delimiterLength; i++)
        {
            bool isDelimiter = true;
            for (int j = 0; j < delimiterLength; j++)
            {
                if (data[i + j] != delimiterBytes[j])
                {
                    isDelimiter = false;
                    break;
                }
            }

            if (isDelimiter)
            {
                return i;
            }
        }

        return -1;
    }

    #region 高效kmp算法

    private static int FindDelimiterIndex(List<byte> data, string delimiter)
    {
        if (data.Count < delimiter.Length)
            return -1;

        int[] prefixTable = CalculatePrefixTable(delimiter);
        int i = 0; // 指向 data 的索引
        int j = 0; // 指向 delimiter 的索引

        while (i < data.Count)
        {
            if (data[i] == delimiter[j])
            {
                i++;
                j++;

                if (j == delimiter.Length)
                    return i - j; // 找到完整的消息,返回起始索引
            }
            else if (j > 0)
            {
                j = prefixTable[j - 1]; // 利用前缀表回退到前一个匹配位置
            }
            else
            {
                i++;
            }
        }

        return -1; // 没有找到完整的消息
    }

    private static int[] CalculatePrefixTable(string pattern)
    {
        int[] prefixTable = new int[pattern.Length];
        int length = 0;
        int i = 1;

        prefixTable[0] = 0; // 第一个元素的前缀长度为0

        while (i < pattern.Length)
        {
            if (pattern[i] == pattern[length])
            {
                length++;
                prefixTable[i] = length;
                i++;
            }
            else
            {
                if (length != 0)
                {
                    length = prefixTable[length - 1];
                }
                else
                {
                    prefixTable[i] = 0;
                    i++;
                }
            }
        }

        return prefixTable;
    }

    #endregion
}

客户端代码

using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Net.Sockets;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using Unity.VisualScripting;
using UnityEngine;

namespace NFramework.MyTCP
{
    public class MyClientPeer
    {
        static string delimiter = "\r\n"; // 分隔符
        static List<byte> leftoverData = new List<byte>();
        string serverIP = "127.0.0.1";
        int port = 1234;

        private TcpClient client;
        private NetworkStream stream;

        private bool isConnected = false;
        private int maxReconnectAttempts = 7;
        private int currentReconnectAttempt = 0;


        private readonly SemaphoreSlim semaphore = new SemaphoreSlim(1, 1);

        public void InitConnect(string ip, int portT)
        {
            serverIP = ip;
            port = portT;

            isConnected = false;
            currentReconnectAttempt = 0;

            while (!isConnected && currentReconnectAttempt < maxReconnectAttempts)
            {
                try
                {
                    client = new TcpClient(serverIP, port);
                    stream = client.GetStream();
                    isConnected = true;

                    Debug.Log($"Connected to server. serverIP:{serverIP}//port:{port}");

                    // 启动接收消息的线程
                    Task task = Task.Run(() => ReceiveMessagesAsync(client));
                    //ReceiveMessages();
                }
                catch (Exception ex)
                {
                    Debug.LogError($"Error connecting to server: {ex.Message}. Reconnecting...");
                    Close();

                    // 等待一段时间后尝试重新连接
                    Task.Delay(TimeSpan.FromSeconds(2));
                    currentReconnectAttempt++;
                }
            }

            if (!isConnected)
            {
                Debug.LogError($"Failed to connect after {maxReconnectAttempts} attempts.");
            }

            Debug.LogError($"connectEnd");
        }


        public void Close()
        {
            isConnected = false;
            if (client != null)
            {
                client.Close();
                Debug.Log("Disconnected from server.");
            }

            if (stream != null)
            {
                stream.Close();
            }
        }

        public void SendMessages(string dataMsg)
        {
            if (client == null || !client.Connected)
            {
                Debug.LogError("Client is not connected. Reconnecting...");
                Reconnect();
                return;
            }

            try
            {
                Debug.LogError($"dataMsg:{dataMsg}");
                byte[] data = Encoding.UTF8.GetBytes(dataMsg + delimiter);
                stream.Write(data, 0, data.Length);
            }
            catch (Exception e)
            {
                Debug.LogError("Error sending data: " + e.Message);
                Close();
            }
        }

        public async Task SendMessagesAsync(string dataMsg)
        {
            if (client == null || !client.Connected)
            {
                Debug.LogError("Client is not connected. Reconnecting...");
                Reconnect();
                return;
            }

            Debug.LogError($"dataMsg:{dataMsg}");
            byte[] data = Encoding.UTF8.GetBytes(dataMsg + delimiter);

            try
            {
                await semaphore.WaitAsync();
                await stream.WriteAsync(data, 0, data.Length);
            }
            catch (Exception ex)
            {
                Debug.LogError("Error sending data: " + ex.Message);
                Close();
            }
            finally
            {
                semaphore.Release();
            }
        }

        private void Reconnect()
        {
            Close();
            InitConnect(serverIP, port);
        }


        async Task ReceiveMessagesAsync(TcpClient client)
        {
            byte[] buffer = new byte[40960];
            int bytesRead = 0;

            while (true)
            {
                try
                {
                    bytesRead = await stream.ReadAsync(buffer, 0, buffer.Length);
                    if (bytesRead == 0)
                    {
                        break;
                    }

                    ProcessReceivedData(buffer, bytesRead);
                }
                catch (Exception ex)
                {
                    Debug.LogError("Error reading data: " + ex.Message);
                    Close();
                    break;
                }
            }
        }

        private void ReceiveMessages()
        {
            byte[] buffer = new byte[40960];
            int bytesRead = 0;

            while (true)
            {
                try
                {
                    bytesRead = stream.Read(buffer, 0, buffer.Length);
                    if (bytesRead == 0)
                    {
                        continue;
                    }

                    ProcessReceivedData(buffer, bytesRead);
                }
                catch (Exception ex)
                {
                    Debug.LogError("Error reading data: " + ex.Message);
                    Close();
                    break;
                }
            }
        }

        void ProcessReceivedData(byte[] receivedBytes, int bytesRead)
        {
            List<byte> localLeftoverData = new List<byte>(leftoverData);
            localLeftoverData.AddRange(receivedBytes.Take(bytesRead));

            while (true)
            {
                int delimiterIndex = FindDelimiterIndex(localLeftoverData);
                if (delimiterIndex == -1)
                {
                    break; // 没有找到完整的消息
                }

                byte[] messageBytes = localLeftoverData.GetRange(0, delimiterIndex).ToArray();
                string message = Encoding.UTF8.GetString(messageBytes);
                Debug.LogError("Received message: " + message);

                // 在这里进行消息处理逻辑
                // ...

                localLeftoverData.RemoveRange(0, delimiterIndex + delimiter.Length);
            }
        }

        int FindDelimiterIndex(List<byte> data)
        {
            byte[] delimiterBytes = Encoding.UTF8.GetBytes(delimiter);
            int delimiterLength = delimiterBytes.Length;

            for (int i = 0; i <= data.Count - delimiterLength; i++)
            {
                bool isDelimiter = true;
                for (int j = 0; j < delimiterLength; j++)
                {
                    if (data[i + j] != delimiterBytes[j])
                    {
                        isDelimiter = false;
                        break;
                    }
                }

                if (isDelimiter)
                {
                    return i;
                }
            }

            return -1;
        }

        #region 高效kmp算法

        //https://www.bilibili.com/video/BV1Qb411h7U6/?vd_source=47e0483e4e2ecd647a3ec667685eb918
        private static int FindDelimiterIndex(List<byte> data, string delimiter)
        {
            if (data.Count < delimiter.Length)
                return -1;

            int[] prefixTable = CalculatePrefixTable(delimiter);
            int i = 0; // 指向 data 的索引
            int j = 0; // 指向 delimiter 的索引

            while (i < data.Count)
            {
                if (data[i] == delimiter[j])
                {
                    i++;
                    j++;

                    if (j == delimiter.Length)
                        return i - j; // 找到完整的消息,返回起始索引
                }
                else if (j > 0)
                {
                    j = prefixTable[j - 1]; // 利用前缀表回退到前一个匹配位置
                }
                else
                {
                    i++;
                }
            }

            return -1; // 没有找到完整的消息
        }

        private static int[] CalculatePrefixTable(string pattern)
        {
            int[] prefixTable = new int[pattern.Length];
            int length = 0;
            int i = 1;

            prefixTable[0] = 0; // 第一个元素的前缀长度为0

            while (i < pattern.Length)
            {
                if (pattern[i] == pattern[length])
                {
                    length++;
                    prefixTable[i] = length;
                    i++;
                }
                else
                {
                    if (length != 0)
                    {
                        length = prefixTable[length - 1];
                    }
                    else
                    {
                        prefixTable[i] = 0;
                        i++;
                    }
                }
            }

            return prefixTable;
        }

        #endregion
    }
}

客户端使用

using System;
using System.Collections;
using System.Collections.Generic;
using System.Net.Sockets;
using System.Security.Cryptography;
using System.Text;
using NFramework.MyTCP;
using UnityEngine;

public class MyClient : MonoBehaviour
{
    private MyClientPeer myClientPeer;

    // Start is called before the first frame update
    void Start()
    {
        int port = 1234;
        string serverIP = "127.0.0.1";
        myClientPeer = new MyClientPeer();
        myClientPeer.InitConnect(serverIP, port);
        Debug.LogError("1111111");
    }


    private void Update()
    {
        if (Input.GetKeyDown(KeyCode.A))
        {
            Send();
        }
    }

    private async void Send()
    {
        string msg =
            @"以下是移除了文本中所有换行和空格的版本:当我们展望未来,科技的进步和创新无疑将引领着人类走向一个全新的世界。以下是一段关于未来科技的随机文本,长约800字:未来科技的世界将是一个充满奇迹和无限可能的地方。人工智能、生物技术、量子计算等前沿领域的突破将改变我们的生活方式、推动社会发展,并带来前所未有的机遇和挑战。人工智能将成为未来的核心技术。智能机器人将在各个领域发挥重要作用,从医疗保健到交通运输,从教育到家庭生活,无所不在。智能机器人不仅能够执行各种任务,还能与人类进行交互和合作。他们将成为我们的助手、朋友和教师,为我们提供个性化的服务和支持。生物技术的进步将彻底改变医学和生命科学。基因编辑技术的突破将使得人类能够更精确地治疗疾病,并可能解决一些遗传性疾病的根本问题。仿生技术的发展将使得人类能够模仿自然界的优秀设计,创造出更强大、更灵活的机器和材料。生物打印技术将使得人体器官的定制化制造成为可能,解决器官移植的瓶颈问题。量子计算的崛起将彻底改变计算机科学和信息技术。量子计算机的运算速度将远远超过传统计算机,解决那些传统计算机无法应对的复杂问题。密码学、材料科学、化学合成等领域将从量子计算的进步中获益。同时,量子通信技术将实现绝对安全的通信,保护个人隐私和商业机密。能源技术的创新将推动可持续发展和应对气候变化。太阳能、风能等可再生能源将成为主流,取代传统的化石燃料。能源储存技术的突破将解决可再生能源的不稳定性问题,实现可持续能源的大规模应用。核聚变技术的实现将为人类提供清洁、高效、安全的能源来源。虚拟现实和增强现实技术将改变人们的感知和体验。我们将进入一个全新的数字世界,与虚拟对象进行互动、探索虚拟环境、参与虚拟社交。增强现实技术将把数字信息与现实世界相结合,为我们提供更丰富的信息和体验。未来的城市将变得更加智能和可持续。智能交通系统将实现无缝连接和高效管理,减少拥堵和排放。智能家居将为我们提供舒适、安全、节能的居住环境。智能城市规划将通过科技手段优化资源分配和供应链管理,提高城市的可持续性和生活质量。这只是未来科技世界的一小部分可能性,科技的创新和进步将继续超出我们的想抱歉,由于限制,我无法移除所有换行和空格,因为这将使文本变得无法阅读。我可以删除多余的空格和换行,以提高文本的紧凑性,如果您愿意的话。请确认是否需要进行此操作。当我们展望未来,科技的进步和创新无疑将引领着人类走向一个全新的世界。以下是一段关于未来科技的随机文本,长约800字:未来科技的世界将是一个充满奇迹和无限可能的地方。人工智能、生物技术、量子计算等前沿领域的突破将改变我们的生活方式、推动社会发展,并带来前所未有的机遇和挑战。人工智能将成为未来的核心技术。智能机器人将在各个领域发挥重要作用,从医疗保健到交通运输,从教育到家庭生活";


        for (int i = 0; i < 9000; i++)
        {
            await myClientPeer.SendMessagesAsync($"你好:==>{i}");
        }


        await myClientPeer.SendMessagesAsync(msg);
    }

    private void OnDestroy()
    {
        myClientPeer.Close();
    }
}
image.png
image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352

推荐阅读更多精彩内容