简介
粘包 / 拆包问题是网络比较底层的问题,在数据链路层、网络层以及传输层都有可能发生。我们日常的网络应用开发大都在传输层进行,由于UDP有消息保护边界,不会发生粘包拆包问题,因此粘包拆包问题只发生在TCP协议中。
什么是粘包 / 拆包?
假设客户端向服务端连续发送了两个数据包,用packet1和packet2来表示,那么服务端收到的数据可以分为三种,现列举如下:
- 第一种情况,接收端正常收到两个数据包,即没有发生拆包和粘包的现象,此种情况不在本文的讨论范围内。
- 第二种情况,接收端只收到一个数据包,由于TCP是不会出现丢包的,所以这个数据包中包含了发送端发送的两个数据包的信息,这种现象即为粘包。这种情况由于接收端不知道这两个数据包的界限,所以对于接收端来说很难处理。
- 第三种情况,这种情况有两种表现形式,如下图。接收端收到了两个数据包,但是这两个数据包要么是不完整的,要么就是多出来一块,这种情况即发生了拆包和粘包。这两种情况如果不加特殊处理,对于接收端同样是不好处理的。
发生粘包 / 拆包的原因
发生TCP粘包 / 拆包主要由于以下原因:
-
应用程序写入的数据大于套接字缓冲区,将会发生拆包。
-
应用程序写入数据小于套接字缓冲区,网卡将应用多次写入的数据发送到网络上,将会发生粘包。
-
进行MSS(最大报文长度)大小的TCP分段,当(TCP报文长度 - TCP头部长度)> MSS 的时候将发生拆包。
-
接收方法不及时读取套接字缓冲区中的数据,将发生粘包。
解决方法
TCP本身是面向流的,作为网络服务器,如何这源源不断涌来的数据流中拆分出或者合并出有意义的信息呢?通常会有以下常用的方法:
1、包首部。发送端给每个数据包添加包首部,首部中应该至少包含数据包的长度,这样接收端在接收到数据后,通过读取包首部的长度字段,便知道每一个数据包的实际长度了。
2、固定长度。发送端将每个数据包封装为固定长度(不够的可以通过补0填充),这样接收端每次从接收缓冲区中读取固定长度的数据就自然而然的把每个数据包拆分开来。
3、设置边界。可以在数据包之间设置边界,如添加特殊符号,这样,接收端通过这个边界就可以将不同的数据包拆分开。