C++ 结构体内存对齐

定义了两个结构体

cpp 复制代码
 typedef struct Cmd {
    uint8_t ua;
    uint8_t ub;
    uint8_t uc;

    uint32_t ue;
} Cmd_t;


typedef struct Cmd_tag {
    uint8_t value;
    uint8_t data[1]; // 将 data 定义为指向 Cmd_t 结构体的指针
} tag_t;

在实际使用中,看见前人的代码是,new 一块内存放两个结构体。我对内存大小有些疑惑。代码如下

cpp 复制代码
void test2()
{
    int size = 9;
    unsigned char* pdata = new unsigned char[size];
    memset(pdata, 0, size);
    // 解释 pdata 为 tag_t* 类型的指针
    tag_t* pCmd = (tag_t*)pdata;
    pCmd->value = 1;
    // 将 pdata + 1 解释为 Cmd_t* 类型的指针,并将其赋给 data 成员
    Cmd_t*  p= (Cmd_t*)(pdata + 1);

    p->ua = 10;
    p->ub = 10;
    p->uc = 10;
    p->ue = 10;


  
    std::cout << "value: " << static_cast<int>(pCmd->value) << std::endl;
    std::cout << "Address of pCmd->value: " << static_cast<void*>(&pCmd->value) << std::endl;

    std::cout << "data->a: " << static_cast<int>(p->ua) << std::endl;
    std::cout << "Address of p->a: " << static_cast<void*>(&p->ua) << std::endl;

    std::cout << "data->ub: " << static_cast<int>(p->ub) << std::endl;
    std::cout << "Address of p->ub: " << static_cast<void*>(&p->ub) << std::endl;

    std::cout << "data->uc: " << static_cast<int>(p->uc) << std::endl;
    std::cout << "Address of p->uc: " << static_cast<void*>(&p->uc) << std::endl;

    std::cout << "data->ue: " << p->ue << std::endl;
    std::cout << "Address of p->ue: " << static_cast<void*>(&p->ue) << std::endl;

    delete[] pdata;
}

输出如下

解释:

cpp 复制代码
 std::cout << "Size of tag_t: " << sizeof(tag_t) << " bytes" << std::endl;
 std::cout << "Size of Cmd_t: " << sizeof(Cmd_t) << " bytes" << std::endl;
// Size of tag_t: 2 bytes
 //Size of Cmd_t: 8 bytes

结构体 tag_t需要2 bytes

结构体 Cmd_t 需要 8 bytes

一开始我以为,new一块内存存放她两需要 2+8 =10 bytes

但是由于 将 pdata + 1 解释为 Cmd_t* 类型的指针,并将其赋给 data 成员

cpp 复制代码
Cmd_t*  p= (Cmd_t*)(pdata + 1);

相当于 Cmd_tag.data 和 Cmd_t.ua公用一个字节的内存

所以new的时候只需要 9 bytes

.还有一个坑:

最开始计算Cmd_t大小时 ,我以为是7bytes. 后来发现C++ 结构体在这里右4字节对齐

可以观察输出, p->ue的地址比 p->uc的地址增加了 2

所以如果代码里,size =8 程序会崩溃

相关推荐
weixin_472339461 小时前
高效处理大体积Excel文件的Java技术方案解析
java·开发语言·excel
枯萎穿心攻击1 小时前
响应式编程入门教程第二节:构建 ObservableProperty<T> — 封装 ReactiveProperty 的高级用法
开发语言·unity·c#·游戏引擎
Eiceblue3 小时前
【免费.NET方案】CSV到PDF与DataTable的快速转换
开发语言·pdf·c#·.net
tan180°3 小时前
MySQL表的操作(3)
linux·数据库·c++·vscode·后端·mysql
m0_555762903 小时前
Matlab 频谱分析 (Spectral Analysis)
开发语言·matlab
浪裡遊4 小时前
React Hooks全面解析:从基础到高级的实用指南
开发语言·前端·javascript·react.js·node.js·ecmascript·php
彭祥.5 小时前
Jetson边缘计算主板:Ubuntu 环境配置 CUDA 与 cudNN 推理环境 + OpenCV 与 C++ 进行目标分类
c++·opencv·分类
lzb_kkk5 小时前
【C++】C++四种类型转换操作符详解
开发语言·c++·windows·1024程序员节
好开心啊没烦恼5 小时前
Python 数据分析:numpy,说人话,说说数组维度。听故事学知识点怎么这么容易?
开发语言·人工智能·python·数据挖掘·数据分析·numpy
简佐义的博客6 小时前
破解非模式物种GO/KEGG注释难题
开发语言·数据库·后端·oracle·golang