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);
注意:
'The expected size of bytes from reply is 1108, which is correct'。不,不是。没有'expected size'。 TCP 是一种流式传输协议。如果 recv()
returns 为正数,则传输的字节数可以是从 1 到提供的缓冲区长度的任何值。您必须编写 loop 代码才能确保准确获得 N 字节。
在假设为正之前,还必须检查read_result
是否为零,这表明对端已断开连接。不要省略这一步。
strlen(server_reply)
无关紧要。您根本不知道是否存在尾随空值。可能根本没有空值,也可能有多个空值,包括在本例中,一开始就有一个空值。
我正在编写一个客户端,以在 LINUX.
上使用 c 脚本在 TCP 下通过套接字读取从服务器发回的数据服务器永远是 运行,我验证我是否可以得到 netcat localhost [PORT_NUMBER]
的回复。使用 netstat -nap
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);
注意:
'The expected size of bytes from reply is 1108, which is correct'。不,不是。没有'expected size'。 TCP 是一种流式传输协议。如果
recv()
returns 为正数,则传输的字节数可以是从 1 到提供的缓冲区长度的任何值。您必须编写 loop 代码才能确保准确获得 N 字节。在假设为正之前,还必须检查
read_result
是否为零,这表明对端已断开连接。不要省略这一步。strlen(server_reply)
无关紧要。您根本不知道是否存在尾随空值。可能根本没有空值,也可能有多个空值,包括在本例中,一开始就有一个空值。