zoukankan      html  css  js  c++  java
  • C# Socket通信 定义收发信息协议处理粘包

    客户端发送数据到服务端时可能有以下三种情况:

    1.服务端完整接收客户端发送的一条消息;
    2.客户端发送的一条消息被当成两条消息处理;
    3.客户端发送的两条消息(甚至更多)被合并成一条消息接收;
    原因是NetWrokStream写入数据时,数据并没有立即发往远程主机,而是保存在了TCP缓存(TCP Buffer)中,经过一段时间之后再进行发送,对于传输二进制文件并没什么影响,但对于文本来说,就需要明确发送文本的边界。

    // 协议:[length=XX]..... 其中XX是字符串的长度(注意不是字节数组buffer的长度)

    如:"[length=12]Hello World!"

    • 处理逻辑

    服务端接收字符串后可能有下面两种情况:

    1."[""]"中括号是完整的,可以读取到字符串的length长度,然后根据这个数值与后面的字符串长度相比,如果相等则说明接收到的数据是完整的;如果多了,则说明接收的字节数多了,取出合适的长度,并将剩余的数据进行缓存,等待下一次接收数据的时候进行合并;如果少了,说明接收数据不完整,将数据进行缓存,等待下一次接收的户数的时候进行合并。

    2."[""]"中括号本身不完整,此时读取不到字符串的length长度,将接收的数据进行缓存,等待读取下次接收的数据,然后将两次合并字后的数据按上面的方式进行处理。

    • 编码实现

    定义RequestHandler类用于服务端解析客户端发送的字符串数据:

        public class RequestHandler
        {
            private string temp = string.Empty;
    
            public string[] GetActualString(string input) {
                return GetActualString(input, null);
            }
    
            public string[] GetActualString(string input, List<string> outputList) {
                if (outputList == null) {
                    outputList = new List<string>();
                }
                if (!string.IsNullOrEmpty(temp)) {
                    input = temp + input;
                }
    
                string output;
                // (?<=patten) 反向肯定预查询   (?=patten) 正向肯定预查询
                string patten = @"(?<=^[length=)(d+)(?=])";
                int length;
    
                if (Regex.IsMatch(input, patten)) {
                    Match m = Regex.Match(input, patten);
                    // 获取消息字符串实际长度
                    length = Convert.ToInt32(m.Groups[0].Value);
                    // 获取需要进行截取的位置
                    int startIndex = input.IndexOf(']') + 1;
                    // 获取从截取位置开始后所有字符的长度
                    output = input.Substring(startIndex);
    
                    if (output.Length == length) {
                        // output长度与消息字符串长度相等,说明刚好是完整的一条消息
                        outputList.Add(output);
                        temp = "";
                    }
                    else if (output.Length < length) {
                        // output长度小于消息字符串长度,说明消息没有发完整
                        // 应将整条消息,包括元数据全部缓存,与下一条数据合并起来再进行处理
                        temp = input;                    
                    }
                    if (output.Length > length) {
                        // output长度大于消息字符串长度,说明消息发完整了,但是有多余的数据
                        // 多余的数据可能是截断消息,也可能是多条完整的消息
                        // 截取字符串
                        output = output.Substring(0, length);
                        outputList.Add(output);
                        temp = "";
                        // 缩短input的长度
                        input = input.Substring(startIndex + length);                    
                        // 递归调用
                        GetActualString(input, outputList);                    
                    }
                }
                else {
                    // "[","]"不完整
                    temp = input;
                }
                return outputList.ToArray();
            }
        }
  • 相关阅读:
    >动态规划 4.26
    树链剖分+线段树求路径交
    PTA团体程序设计天梯赛-练习集 L2 网红点打卡攻略(模拟)
    PTA团体程序设计天梯赛-练习集 L2完全二叉树的层序遍历(递归)
    PTA团体程序设计天梯赛-练习集 L3-020 至多删三个字符 (dp)
    codeforces1509 D. Binary Literature (构造+指针)
    函数内容小结
    关于vim复制剪贴粘贴命令的总结-转
    GCC编译命令常用选项
    Ubuntu 和 windows1下文件夹共享的指令
  • 原文地址:https://www.cnblogs.com/zhengzc/p/11208377.html
Copyright © 2011-2022 走看看