arrays - 在 C 中使用 UDP 从客户端传输布尔数组
问题描述
我需要创建一个简单的 modbus 应用程序,它将以 bool 类型传输数据。我为此创建了客户端和服务器代码。客户端:
int Client(bool message[8])
{
struct sockaddr_in si_other;
int s, slen=sizeof(si_other);
bool buf[BUFLEN];
WSADATA wsa;
if (WSAStartup(MAKEWORD(2,2),&wsa) != 0)
{
exit(EXIT_FAILURE);
return 1;
}
if ( (s=socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) == SOCKET_ERROR)
{
exit(EXIT_FAILURE);
return 2;
}
memset((bool *) &si_other, 0, sizeof(si_other));
si_other.sin_family = AF_INET;
si_other.sin_port = htons(PORT);
si_other.sin_addr.S_un.S_addr = inet_addr(SERVER);
if (sendto(s, message, strlen(message) , 0 , (struct sockaddr *) &si_other, slen) == SOCKET_ERROR)
{
exit(EXIT_FAILURE);
return 3;
}
// closesocket(s);
// WSACleanup();
return 0;
}
服务器端:
int main()
{
SOCKET s;
struct sockaddr_in server, si_other;
int slen , recv_len;
bool buf[BUFLEN];
WSADATA wsa;
slen = sizeof(si_other) ;
printf("\nInitialising Winsock...");
if (WSAStartup(MAKEWORD(2,2),&wsa) != 0)
{
printf("Failed. Error Code : %d",WSAGetLastError());
exit(EXIT_FAILURE);
}
printf("Initialised.\n");
if((s = socket(AF_INET , SOCK_DGRAM , 0 )) == INVALID_SOCKET)
{
printf("Could not create socket : %d" , WSAGetLastError());
}
printf("Socket created.\n");
server.sin_family = AF_INET;
server.sin_addr.s_addr = INADDR_ANY;
server.sin_port = htons( PORT );
if( bind(s ,(struct sockaddr *)&server , sizeof(server)) == SOCKET_ERROR)
{
printf("Bind failed with error code : %d" , WSAGetLastError());
exit(EXIT_FAILURE);
}
puts("Bind done");
while(1)
{
printf("Waiting for data...\n");
fflush(stdout);
memset(buf,'0', BUFLEN);
if ((recv_len = recvfrom(s, buf, BUFLEN, 0, (struct sockaddr *) &si_other, &slen)) == SOCKET_ERROR)
{
printf("recvfrom() failed with error code : %d" , WSAGetLastError());
exit(EXIT_FAILURE);
}
printf("Received packet from %s:%d\n", inet_ntoa(si_other.sin_addr), ntohs(si_other.sin_port));
for (int i=0; i<=7; i++) {
printf("%d", buf[i]);
}
printf("Data: %d\n" , buf);
printf("%s-%s-%s\n",buf[0],buf[1],buf[2]); //When I run this code, it works just like the code with 'for' and gives an error.
if (sendto(s, buf, recv_len, 0, (struct sockaddr*) &si_other, slen) == SOCKET_ERROR)
{
printf("sendto() failed with error code : %d" , WSAGetLastError());
exit(EXIT_FAILURE);
}
}
closesocket(s);
WSACleanup();
return 0;
}
当我运行应用程序时,数据传输不正确。例如,当我发送数据为“101010”时,它传输“100000”;如果发送数据为“110101”,则发送“110000”,如果发送为“011111”,则发送“000000”。换句话说,为了正确读取值为“1”的数据,该数据之前的所有值都必须为“1”。
当我删除代码中的 'for' 部分并尝试直接读取整个 buf ( printf("Data: %d\n" , buf) )时,它会读取 '6421972' 数据。即使我改变数据,这个数字也不会改变。我能做些什么来克服这个问题?
解决方案
在您的客户端中, strlen(message) 将计算字符,直到第一个 '0' 出现。因此,您发送的 bool 数组的长度永远不会是 8。
然后,您的客户端必须在参数中也采用缓冲区长度来解决此问题。
如果您的缓冲区是一个真正的 C 数组(不是分配的指针),那么sizeof语句可以给出长度。但是如果你对 buf 使用 malloc 的指针,sizeof 语句将始终返回 8,永远不会更少,永远不会更多(在 64 位系统中),这只是指针的大小。
简而言之,始终在缓冲区旁边保留一个长度整数。
推荐阅读
- apache-kafka - 如何在kafka流中动态处理并发送到不同的主题
- validation - Ghostscript 作为 PDF 验证器:命令和结果
- java - 为什么这个递归循环等于 200?
- r - 如何计算 XML 文件中的数字(字符串格式)
- java - 如何为具有 yolo 输出层的多任务学习网络准备数据集?
- mysql - St_within 用于获取 10 英里半径的 MBR mysql
- php - 如何根据php中列的最大值从数组对象中获取数组对象
- sql-server - 在 SQL Server 中进行正向还原是否安全?
- javascript - Flutter - 在 null 上调用了“then”方法
- windows - SourceTree 在版本更新后不会自动刷新