c - 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 的编码风格。
解决方案
这里没有问题的证据,只有错误的代码。如果recv()
返回 1088,它肯定会将 1088 字节传输到缓冲区中。显然,接收到的数据以空字节开始。要正确打印,请使用
printf("%.*s\n", read_result, server_reply);
注意:
'回复的预期字节大小是 1108,这是正确的'。不,不是。没有“预期大小”。TCP 是一种流协议。如果
recv()
返回一个正数,则传输的字节数可以是从 1 到提供的缓冲区长度的任何值。您必须编写一个循环以确保准确获取N个字节。在假设它为正之前,您还必须检查
read_result
是否为零,这表明对等方已断开连接。不要省略这一步。strlen(server_reply)
无关紧要。您根本不知道是否有尾随空值。可能根本没有空值,也可能有几个,包括在这种情况下,一开始就有一个。
推荐阅读
- c# - 访问数组的元素
- quarkus - hibernate ORM - 缓存查询
- macos - 如何在 OSX Catalina (10.15) 中使用 Wireshark 捕获 USB 流量
- python - 每小时随机播放一次“PASSWORD-HASHERS” Django
- javascript - 如何在同一页面上多次重复使用我的 javascript 滑块?
- postgresql - 是否可以在排除约束中调用函数以仅返回特定列
- amazon-web-services - CloudWatch 重启警报操作未能执行
- objective-c - 如何从包含嵌套 JSON 的 NSDictionary 中提取 NSString
- python - 有什么方法可以在 python 中解码 str 吗?AttributeError:“str”对象没有属性“decode”
- character-encoding - 为什么某些字符(例如“ç”)看起来与其他字符不同?