我正在尝试接收TCP套接字的消息并将其存储在uint8_t
数组中。
我要接收的缓冲区长度为8个字节,包含4个唯一值。
字节1:值1是uint8_t,字节2-3:值2是uint16_t,字节4:值3是uint8_t,字节5-8:值4是无符号长。
Endianness是大端序。
int numBytes = 0;
uint8_t buff [8];
if ((numBytes = recv(sockfd, buff, 8, 0)) == -1)
{
perror("recv");
exit(1);
}
uint8_t *pt = buff;
printf("buff[0] = %u\n", *pt);
++pt;
printf("buff[1] = %u\n", *(uint16_t*)pt);
但第二个printf
打印出一个意外的值。我做错了什么来提取两个字节或者我的打印功能有问题吗?
答案 0 :(得分:2)
一旦您的数据到达缓冲区,您就有2个问题需要处理。
第一个是遵守别名规则,这是通过仅向char*
投射非char类型指针来实现的,因为char
可以别名任何。你应该从不将char*
转换为非char类型的指针。
第二种是遵守网络字节排序协议,通过网络传输的整数在传输之前转换为网络顺序,并在接收后从网络顺序转换。为此,我们通常使用htons
,htonl
,ntohs
和ntohl
。
这样的事情:
// declare receive buffer to be char, not uint8_t
char buff[8];
// receive chars in buff here ...
// now transfer and convert data
uint8_t a;
uint16_t b;
uint8_t c;
uint32_t d;
a = static_cast<uint8_t>(buff[0]);
// always cast the receiving type* to char*
// never cast char* to receiving type*
std::copy(buff + 1, buff + 3, (char*)&b)
// convert from network byte order to host order
b = ntohs(b); // short version (uint16_t)
c = static_cast<uint8_t>(buff[3]);
std::copy(buff + 4, buff + 8, (char*)&d)
d = ntohl(d); // long version (uint32_t)
答案 1 :(得分:1)
也许就像这样(big-endian)
uint8_t buff [8];
// ...
uint8_t val1 = buff[0];
unit16_t val2 = buff[1] * 256 + buff[2];
unit8_t val3 = buff[3];
unsigned long val4 = buff[4] * 16777216 + buff[5] * 65536 + buff[6] * 256 + buff[7];