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

时间:2018-05-09 00:27:27

标签: c linux sockets tcp tcp-ip

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

服务器永远在运行,我验证是否可以获得netstat -nap的回复。使用#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; }

进行检查时,服务器处于LISTEN状态

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

这是代码。

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

bash中的打印结果是

int read_result = recv(socket_desc, server_reply, msg_rep_size, 0); 

如果我从

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

{{1}}

感谢您的帮助。我是套接字编程的新手,无法跟踪发生了什么。

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

PS2。请随意评论低级别c的编码风格。

1 个答案:

答案 0 :(得分:2)

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

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

NB:

  1. &#39;来自回复的预期字节大小为1108,这是正确的&#39;。不,它不是。没有预期的尺寸&#39;。 TCP是一种流媒体协议。如果recv()返回正数,则传输的字节数可以是从1向上到提供的缓冲区长度。您必须编写循环以确保获得 N 字节。

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

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