0

0

c++如何解析二进制协议报文_c++ 结构体映射、对齐与字节序处理【方法】

尼克

尼克

发布时间:2026-01-15 15:49:32

|

128人浏览过

|

来源于php中文网

原创

结构体直接映射二进制报文会出错,因默认对齐和字节序不匹配协议要求;需用__attribute__((packed))禁用填充,并用ntohs等函数手动转换字节序,同时推荐memcpy逐字段解析以避免未定义行为。

c++如何解析二进制协议报文_c++ 结构体映射、对齐与字节序处理【方法】

结构体直接映射二进制报文会出错,因为默认对齐和字节序不匹配

绝大多数二进制协议(如自定义 TCP/UDP 报文、CAN 帧、设备固件升级包)要求字段按固定偏移、固定长度、固定字节序(通常是大端)排列。C++ struct 默认受编译器对齐规则影响,sizeof 往往大于字段字节和,且成员起始地址不等于协议文档写的 offset;同时 x86/x64 是小端机,直接读 uint16_t 会把高位字节当低位,导致数值错乱。

#pragma pack(1) 强制关闭结构体填充

这是最常用也最易上手的方案,让每个字段紧挨着前一个字段存储,消除隐式 padding:

struct __attribute__((packed)) ModbusRequest {
    uint8_t  addr;
    uint8_t  func;
    uint16_t reg_start; // 协议里是 big-endian,但这里只是占位
    uint16_t reg_count;
};

注意:#pragma pack(1) 在 MSVC 下生效,GCC/Clang 推荐用 __attribute__((packed))(如上例)。两者效果一致,但后者更跨平台。别写 pack(2)pack(4) —— 只要不是 1,就可能在某些字段间插入 padding,破坏协议 layout。

  • 字段顺序必须严格按协议字节流顺序声明
  • 不能含虚函数、非 POD 类型(如 std::string)、引用或非平凡构造函数
  • sizeof(ModbusRequest) 必须等于协议规定的总长度(这里是 6 字节)

手动处理字节序:用 ntohs/ntohlbswap_16

结构体映射只解决内存布局,不解决字节序。协议中 reg_start 是网络字节序(大端),而 x86 上 uint16_t 按小端解释。必须在解析后转换:

立即学习C++免费学习笔记(深入)”;

Hotpot AI Background Remover
Hotpot AI Background Remover

Hotpot.ai推出的图片背景移除工具

下载
uint8_t buf[6] = {0x01, 0x03, 0x00, 0x0A, 0x00, 0x01};
ModbusRequest* req = reinterpret_cast<ModbusRequest*>(buf);
req->reg_start = ntohs(req->reg_start); // 0x000A → 10
req->reg_count = ntohs(req->reg_count); // 0x0001 → 1

关键点:

  • ntohs(network to host short)适用于所有 POSIX 系统,Windows 需 #include <winsock2.h></winsock2.h> 并链接 ws2_32.lib
  • 若目标平台是 ARM 大端(少见),ntohs 实际是空操作;但协议层仍应统一调用,保持可移植性
  • 不要对 uint8_t 调用字节序函数——它只有一个字节,无序可言

避免指针强转引发未定义行为的稳妥做法

直接 reinterpret_cast 结构体指针有风险:若 buf 地址未按结构体最大对齐要求对齐(如 uint64_t 要求 8 字节对齐),触发未定义行为(尤其在 ARM 或开启严格别名检查时)。更安全的做法是逐字段 memcpy:

struct ModbusRequest {
    uint8_t addr;
    uint8_t func;
    uint16_t reg_start;
    uint16_t reg_count;
} __attribute__((packed));
<p>ModbusRequest req;
memcpy(&req.addr,  buf + 0, 1);
memcpy(&req.func,  buf + 1, 1);
memcpy(&req.reg_start, buf + 2, 2);
memcpy(&req.reg_count, buf + 4, 2);
req.reg_start = ntohs(req.reg_start);
req.reg_count = ntohs(req.reg_count);

这样完全规避对齐问题,且编译器通常能内联优化为单条 load 指令。如果协议字段多、性能敏感,再考虑用 std::bit_cast(C++20)或带对齐检查的 placement new。

对齐和字节序这两个点只要漏掉一个,解析出来的值就不可信;尤其是嵌入式通信场景,错误常表现为“偶发性数据错乱”,排查起来比逻辑 bug 更耗时。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

890

2023.08.02

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

428

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

201

2025.07.04

css中的padding属性作用
css中的padding属性作用

在CSS中,padding属性用于设置元素的内边距。想了解更多padding的相关内容,可以阅读本专题下面的文章。

170

2023.12.07

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

1336

2023.07.26

查看端口占用情况windows
查看端口占用情况windows

端口占用是指与端口关联的软件占用端口而使得其他应用程序无法使用这些端口,端口占用问题是计算机系统编程领域的一个常见问题,端口占用的根本原因可能是操作系统的一些错误,服务器也可能会出现端口占用问题。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1158

2023.07.27

windows照片无法显示
windows照片无法显示

当我们尝试打开一张图片时,可能会出现一个错误提示,提示说"Windows照片查看器无法显示此图片,因为计算机上的可用内存不足",本专题为大家提供windows照片无法显示相关的文章,帮助大家解决该问题。

827

2023.08.01

windows查看端口被占用的情况
windows查看端口被占用的情况

windows查看端口被占用的情况的方法:1、使用Windows自带的资源监视器;2、使用命令提示符查看端口信息;3、使用任务管理器查看占用端口的进程。本专题为大家提供windows查看端口被占用的情况的相关的文章、下载、课程内容,供大家免费下载体验。

460

2023.08.02

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

24

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 10万人学习

Excel 教程
Excel 教程

共162课时 | 19.5万人学习

PHP基础入门课程
PHP基础入门课程

共33课时 | 2.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号