7

我有将字节数组转换为结构的问题,一些字节被忽略或跳过。

给定以下结构,

typedef struct
{
    uint32_t id;
    uint16_t test;
    uint8_t group;
    uint32_t time;
    uint16_t duration;
    uint8_t a;
    uint8_t b;
    uint8_t c;
    uint16_t d;
    uint16_t e;
    uint8_t status;
    uint8_t x;
    uint8_t y;

} testStruct_t, *PtestStruct_t;

我有一个包含以下测试数据的数组:

uint8_t pBuff = { 0x11 , 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0x88, 0x99, 0xAA, 0xBB, 0xCC, 0xDD, 0xEE, 0xFF, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19 };

铸造按如下方式进行:

PtestStruct_t pStruct = (PtestStruct_t)pBuff;

在结构中的某处,一些字节被跳过或忽略。我不知道为什么。这已在 Visual Studio 2012 和需要进行此测试和调试的 ARM 处理器上进行了测试。

我在这里想念什么?我不相信它与 Endian 有关。它可能是两个测试用例中的编译器,我不知道在最后一种情况下该怎么做。

被跳过/忽略的字节是0x880x14

4

2 回答 2

12

您遇到对齐填充。

uint32_t id;     // offset 0
uint16_t test;   // offset 4
uint8_t group;   // offset 6
uint32_t time;   // offset 7

此处显示的偏移量可能是错误的。编译器可能会在“组”和“时间”之间放置填充以确保“时间”在 4 字节边界上(实际对齐是可配置的)

如果您绝对需要这样的结构,您可以使用#pragma pack

#pragma pack(push, 1)
typedef struct
{
    uint32_t id;
    uint16_t test;
    uint8_t group;
    uint32_t time;
    uint16_t duration;
    uint8_t a;
    uint8_t b;
    uint8_t c;
    uint16_t d;
    uint16_t e;
    uint8_t status;
    uint8_t x;
    uint8_t y;

} testStruct_t, *PtestStruct_t;
#pragma pack(pop)
于 2013-11-15T07:15:54.620 回答
3

编译器可能在您的结构字段之间添加了一些字节以进行对齐。您需要使用包装来防止编译器进行填充 - 这必须明确要求 - 在 GCC 下它的属性((打包)),

例子:

      struct __attribute__((__packed__)) mystruct_A {
      char a;
      int b;
     char c;
     };

对于 Visual Studio,请参阅MSDN

于 2013-11-15T07:19:51.067 回答