如何修复 recv 将多个 TCP 段合并为一个?

How to fix recv merging multiple TCP segments into a single one?

我有两个同伴:爱丽丝和鲍勃。 Bob 正试图向 Alice 发送他已知的点列表。

首先,Bob 告诉 Alice 他有多少个节点,然后他将包含节点信息的 n 字符串发送给 Alice。

这是 Bob 的代码:

if ((send(socket_fd, &(int){ 1 }, sizeof(int), 0)) == -1) { perror("Failed to send"); }
sleep(1); // Give him time to process the rqst

send(socket_fd, &peer_cnt, sizeof(peer_cnt), 0); 
for (int i = 0; i < peer_cnt; ++i) {
    char *peer_str = get_string(&peers[i]);
    printf("Send peer\t:\t%s\n", peer_str);
    send(socket_fd, peer_str, strlen(peer_str), 0); 
    free(peer_str);
}   

这是爱丽丝的密码:

char buf[128] = { 0 };
int n = 0;
recv(peer_sock_fd, &n, sizeof(n), 0);
printf("Number of incoming peers:\t%d\n", n); 

for (int i = 0; i < n; ++i) {
    memset(buf, '[=11=]', sizeof(buf));
    recv(peer_sock_fd, buf, sizeof(buf), 0); 
    puts(buf);
}

例子。如果 Bob 向 Alice 发送 2 字符串,则存在两种情况。 首先,爱丽丝分别获得它们(正如我的意图),例如在这种情况下,Alice 的输出可能是 george:127.0.0.1:5555:michael:127.0.0.1:4444:。 其次,爱丽丝将它们作为单个字符串获取,像这样 george:127.0.0.1:5555:michael:127.0.0.1:4444:

为什么有时会合并传入的字符串?

TCP 是一种流式传输协议,当您调用 send() 时,它会将您的消息一个接一个地放入一个特殊的缓冲区,然后发送。由于您在非常短的时间内调用多个 send() 系统调用,其中许多系统调用被放入同一个缓冲区并作为一条消息发送。有时它们会单独发送,有时会合并,有时只会发送一半消息,这是流协议的默认行为。你应该做的(这就是每个体面的 TCP 程序的实现方式)是分隔你的消息。最好的方法是将消息大小整数作为每条消息的前 4 个字节,然后根据消息的大小分隔消息。我建议您阅读 Beej 的套接字指南 (https://beej.us/guide/bgnet/html/multi/index.html),它解释了这个概念等等。非常适合初学者