首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >如何从uint8_t数组中提取不同大小的值?

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

Stack Overflow用户
提问于 2016-10-30 19:31:31
回答 2查看 2.4K关注 0票数 1

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

代码语言:javascript
运行
复制
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输出了一个意想不到的值。我是否做了一些错误的事情来提取这两个字节,或者我的打印函数有问题?

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2016-10-30 19:53:10

也许像这样(大本营)

代码语言:javascript
运行
复制
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];
票数 1
EN

Stack Overflow用户

发布于 2016-10-30 20:13:12

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

第一种是遵守混叠规则,这是通过将非char类型的指针转换到char*来实现的,因为char可以别名任何东西。您不应该将char*转换为非字符类型的指针。

第二种是遵守网络字节排序协议,其中通过网络传输的整数在传输前被转换为网络顺序,并在收到后从网络顺序转换。为此,我们通常使用htonshtonlntohsntohl

就像这样:

代码语言:javascript
运行
复制
// 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)
票数 2
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/40332770

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档