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++免费学习笔记(深入)”;

Shakespeare
Shakespeare

一款人工智能文案软件,能够创建几乎任何类型的文案。

下载
uint8_t buf[6] = {0x01, 0x03, 0x00, 0x0A, 0x00, 0x01};
ModbusRequest* req = reinterpret_cast(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 并链接 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));

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 更耗时。

相关专题

更多
string转int
string转int

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

315

2023.08.02

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

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

195

2025.06.09

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

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

187

2025.07.04

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

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

131

2023.12.07

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

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

576

2023.07.26

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

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

1099

2023.07.27

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

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

790

2023.08.01

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

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

452

2023.08.02

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

6

2026.01.15

热门下载

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

精品课程

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

共48课时 | 7.2万人学习

Excel 教程
Excel 教程

共162课时 | 11.8万人学习

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

共33课时 | 1.9万人学习

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

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