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

时间:2016-10-30 19:31:32

标签: c++ c arrays pointers tcp

我正在尝试接收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打印出一个意外的值。我做错了什么来提取两个字节或者我的打印功能有问题吗?

2 个答案:

答案 0 :(得分:2)

一旦您的数据到达缓冲区,您就有2个问题需要处理。

第一个是遵守别名规则,这是通过仅向char*投射非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)

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