0

我正在从通过 TCP 套接字连接的测试仪器下载 6kB 的数据。我正在使用 SOCK_STREAM:

if((MySocket=socket(PF_INET,SOCK_STREAM,0))==-1) exit(1);

我没有设置任何缓冲区大小,所以我假设我使用的是默认值。我在接收数据时遇到问题。这是声明:

  if((recv(MySocket, &YRaw[indx], sizeChunk*2, 0))==-1)
     {perror("recv()"); exit(9); }  // where sizeChunk=3000; sizeof(YRaw[0])=2

其中 YRaw 是一个 3000 个对象数组,每个对象是 2 个字节。

我在运行代码后观察到的问题是,在 YRaw 数组的大约 3/4 处,有一个零字符串,这是我将数组初始化为的字符串。看起来接收函数可以正常工作到数组的 3/4,然后跳过一大块 YRaw 数组(保留初始化值),然后再次拾取,并且 YRaw 数组填满。缓冲区中剩余的数据要读取,等于 YRaw 数组中跳过的值的数量。

我怀疑这与系统中的某些缓冲区有关。任何人都可以阐明如何解决这个问题吗?

不知道有没有关系,不过我也设置了TCP_NODELAY,如下图:

// TCP_NODELAY option minimizes latency by instructing system to send small msgs immediately
// Insert setsockopt() before connect() occurs
int StateNODELAY = 1; // turn NODELAY on
if(setsockopt(MySocket, IPPROTO_TCP,TCP_NODELAY,
   (void *) &StateNODELAY, sizeof StateNODELAY)==-1) {perror("setsockopt()"); exit(2); };

最好的问候, gkk

4

2 回答 2

3

recv()不一定填满整个缓冲区。它只是读取任何可用的内容并返回已读取的字节数。

因此,检查返回值以查看有多少数据可用,处理该数据,然后recv()再次调用,直到收到您期望的所有数据。

于 2010-08-28T17:27:40.827 回答
0

您应该明确存储结果recv(即不仅检查它是否为-1)。这是实际读取的字节大小,您需要知道缓冲区中有多少字节是有效的。

除此之外,您应该检查是否可以在不阻塞的情况下读取更多字节,并在阻塞之前读取更多字节。

缓冲区大小限制可以在系统中的任何位置,例如以太网 MTU 甚至在您的发送设备上。

于 2010-08-28T17:29:23.787 回答