read()/recv() 成功但缓冲区未更改且为空

read()/recv() successfully but buffer is unchanged and empty

我正在编写一个客户端,以在 LINUX.

上使用 c 脚本在 TCP 下通过套接字读取从服务器发回的数据

服务器永远是 运行,我验证我是否可以得到 netcat localhost [PORT_NUMBER] 的回复。使用 netstat -nap

检查时服务器处于 LISTEN 状态

recv() 函数 returns 预期的字节数,但缓冲区变为空并且 strlen(buffer) 为 0。我也尝试更改为 read(),但我没想到会有不同的结果,并且显示了同样的问题。

这是代码。

#include <stdio.h>
#include <string.h> //strlen
#include <sys/socket.h>
#include <arpa/inet.h> //inet_addr
#include <unistd.h>
#include <stdlib.h>

int main(int argc, char *argv[])
{
    int socket_desc, msg_rep_size = 1200, msg_size = 100;
    struct sockaddr_in server;
    char message[msg_size], server_reply[msg_rep_size];

    //Create socket
    socket_desc = socket(AF_INET, SOCK_STREAM, 0);
    if (socket_desc == -1)
    {
        printf("Could not create socket");
    }

    server.sin_addr.s_addr = inet_addr("0.0.0.0");
    server.sin_family = AF_INET;
    server.sin_port = htons(31114); // PORT_NUMBER

    //Connect to remote server
    if (connect(socket_desc, (struct sockaddr *)&server, sizeof(server)) < 0)
    {
        printf("connect error");
        return 1;
    }
    printf("Connected");


    while (1)
    { 
    // send some data
    bzero(message, msg_size);
    strncpy(message, "REQUEST\n", msg_size);

    if (send(socket_desc, message, msg_size, 0) < 0)
    {
        printf("Send failed");
        return 1;
    }
    printf("Data Send\n");

    // Receive a reply from the server
    bzero(server_reply, msg_rep_size);
    int read_result = recv(socket_desc, server_reply, msg_rep_size, 0); 

    if (read_result < 0)
    {
        printf("Receive failed\n");
        return 1;
    }
    else
    {
        printf("Reply received\n");
        printf("read_result: %d\n", read_result);
        printf("strlen(server_reply): %d\n", (int)strlen(server_reply));
        printf("Reply: %s\n", server_reply);
    }
    usleep(8);
}

return 0;
}

bash中的打印结果是

Data send
Reply received
read_result: 1108 
strlen(server_reply): 0
Reply:

如果我从

更改会显示相同的结果
int read_result = recv(socket_desc, server_reply, msg_rep_size, 0); 

int read_result = read(socket_desc, server_reply, msg_rep_size);

感谢您的帮助。我是套接字编程的新手,无法跟踪正在发生的事情。

PS。来自回复的预期字节大小是 1108,这是正确的。我打算将最大大小设置为 1200 以确认接收到正确的字节数。

PS2。也请随时评论低级 c 的编码风格。

这里没有证据表明存在问题,只有糟糕的代码。如果 recv() 返回 1088,它肯定将 1088 字节传输到缓冲区中。显然,接收到的数据以空字节开头。要正确打印,请使用

printf("%.*s\n", read_result, server_reply);

注意:

  1. 'The expected size of bytes from reply is 1108, which is correct'。不,不是。没有'expected size'。 TCP 是一种流式传输协议。如果 recv() returns 为正数,则传输的字节数可以是从 1 到提供的缓冲区长度的任何值。您必须编写 loop 代码才能确保准确获得 N 字节。

  2. 在假设为正之前,还必须检查read_result是否为零,这表明对端已断开连接。不要省略这一步。

  3. strlen(server_reply) 无关紧要。您根本不知道是否存在尾随空值。可能根本没有空值,也可能有多个空值,包括在本例中,一开始就有一个空值。