首页 > 解决方案 > read()/recv() 成功但缓冲区未更改且为空

问题描述

我正在编写一个客户端来读取从服务器通过 TCP 下的套接字发送回的数据,并在 LINUX 上使用 c 脚本。

服务器一直在运行,我验证我是否netcat localhost [PORT_NUMBER]已经可以得到回复。检查时服务器处于 LISTEN 状态netstat -nap

recv() 函数返回预期的字节数,但缓冲区变为空并且 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 的编码风格。

标签: clinuxsocketstcptcp-ip

解决方案


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

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

注意:

  1. '回复的预期字节大小是 1108,这是正确的'。不,不是。没有“预期大小”。TCP 是一种流协议。如果recv()返回一个正数,则传输的字节数可以是从 1 到提供的缓冲区长度的任何值。您必须编写一个循环以确保准确获取N个字节。

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

  3. strlen(server_reply)无关紧要。您根本不知道是否有尾随空值。可能根本没有空值,也可能有几个,包括在这种情况下,一开始就有一个。


推荐阅读