如何从uint8_t数组中提取不同大小的值

How to extract values of different sizes from an array of uint8_t?

本文关键字:提取 uint8 数组      更新时间:2023-10-16

我正在尝试接收TCP套接字的消息并将其存储在uint8_t数组中。我要接收的缓冲区是8字节长,包含4个唯一的值。字节1:值1是uint8_t,字节2-3:值2是uint16_t,字节4:值3是uint8_t,字节5-8:值4是unsigned long。端序:大端序

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] = %un", *pt);
++pt;
printf("buff[1] = %un", *(uint16_t*)pt);

但是第二个printf打印出一个意外的值。我是否做了一些不正确的事情来提取两个字节,或者我的打印函数有问题?

一旦你的数据到达缓冲区,你有两个问题要注意。

第一个是遵守别名规则,这是通过只将非字符类型指针强制转换到char*来实现的,因为char可以别名任何。你应该永远不要char*转换为非char类型指针。

二是遵守网络字节排序协议,即通过网络传输的整数在传输前转换为网络顺序,在接收后从网络顺序转换。我们通常使用htonshtonlntohsntohl

像这样:

// 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)

也许像这样(大端)

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];