tcp组包2——如何组包

废话不说了,直接上段代码:

private void DetectPacket()
        {
    
    
            int offset = 0;
            while (m_receiveLen > 0)
            {
    
    
                if (m_receiveLen < 4) break;
                int packetLen = BitConverter.ToInt32(m_receiveBuffer, 0);
                if (m_receiveLen < packetLen + 4)
                {
    
    
                    break;
                }

                Message message = new Message();
                message.m_data = new byte[packetLen];
                Array.Copy(m_receiveBuffer, offset + 4, message.m_data, 0, packetLen);
                m_receiveMessages.Enqueue(message);
                offset += (packetLen + 4);
                m_receiveLen -= (packetLen + 4);
            }

            if (offset > 0 && m_receiveLen > 0)
            {
    
    
                Array.Copy(m_receiveBuffer, offset, m_receiveBuffer, 0, m_receiveLen);
            }
        }

解释每行的意思:
while的主循环的主要目的是,分析接收到的数据包含多少个有效的数据包。直到不能组成一个完整的数据包终止循环。

int offset=0;
这里定义一个包的开始位置。

if (m_receiveLen < 4) break;
如果数据长度小于四个字节,那么也就是说,连包的长度我都没获取到,则直接终止循环,继续接收数据吧。

否则的话,则取出包的长度,这里使用的函数BitConverter.ToInt32方法,从m_receiveBuffer的0位置开始,读取四个字节,就是数据包的长度。
int packetLen = BitConverter.ToInt32(m_receiveBuffer, 0);
if (m_receiveLen < packetLen + 4)
{
break;
}

取到长度之后,判断接收的数据长度是否小于包长度+4,如果小于,则说明还不是一个完整的包,则依然终止循环,继续接收数据吧。

经过前面的两个条件检测之后,如果还能成功。那么说明,接收的数据已经足够一个包体了。则要进行分析包了。
Message message = new Message();
message.m_data = new byte[packetLen];
Array.Copy(m_receiveBuffer, offset + 4, message.m_data, 0, packetLen);
m_receiveMessages.Enqueue(message);
offset += (packetLen + 4);
m_receiveLen -= (packetLen + 4);
这6行代码,分别是:
创建一个我们自定义的消息对象。
对message对象的成员m_data赋值,长度为packetLen。
然后从源数组m_receiveBuffer的offset+4开始处,拷贝到message.m_data的0位置,拷贝长度为packetLen。
然后压入队列。
offset则要进行偏移,偏移距离是包体长度+4个字节。
然后接收的有效数据,减去包体长度,减去4个字节。

然后再判断m_receiveLen是否大于0,如果还大于0则继续while分析包。

ok,分析到此似乎一切都是正常的。
那么while跳出之后,还有两行代码,是看是否有剩余的数据,也就是说,如果我们包体长度是100个字节,而一次接收了120个字节,而下一个包比如150个字节,那么剩余的20个字节则不构成一个整包了。
于是乎,从
100字节开始,将剩余的20个字节向前挪动到m_receiveBuffer的开始位置。
这就是:
if (offset > 0 && m_receiveLen > 0)
{
Array.Copy(m_receiveBuffer, offset, m_receiveBuffer, 0, m_receiveLen);
}
这行代码的意思。

Guess you like

Origin blog.csdn.net/wodownload2/article/details/121349858