C++ 结构体内存方式

2024-06-17 01:08
文章标签 c++ 方式 结构 体内

本文主要是介绍C++ 结构体内存方式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

对于这个问题,主要有两方面的内容。其一,内存优化,其二,存储顺序

一、内存优化

所谓的内存优化是结构体成员的存储按成员中数据宽度最宽的成员对齐的一种内存对齐方式。这样做的最大好处就是可以提高处理速度(感兴趣的可以查一下资料),但会增大存储空间,这种优化是对时间和空间的一种权衡。

  • 如果结构体中数据成员的最大宽度为32位的,即4字节,那么该结构体的大小将是4字节的倍数,而不是将所有成员宽度加起来的大小。同样,如果数据成员中最大宽度为64位的,那么改及饿哦固体大小将是8字节的倍数。

    • 猜想一下,下里两个结构的大小是多少?可以输出一下sizeof的结果

      //例 1
      struct str1
      {
      int  a;     //占用4字节,有效内容4字节
      char b;     //占用4字节,有效内容1字节
      };
      struct str2
      {
      double a;
      char  b;        //占用8字节,有效内容1字节
      };
      //初始化代码
      str1 str_1 = { 1024,'A' };
      str2 str_2 = { 10.5,'A' };
    • 答案是:str1的大小为8,str2的大小为16。下图是我截取的内存图:


    这里写图片描述

    第一行为str_2.a的内存
    第二行为str_2.b的内存,cc均为栈区的填充内容
    第四行前4字节为str_1.a的内存
    第四行后4字节为str_1.b的内存

  • 那这样的话岂不是结构体大小为数据成员中最大数据宽度*数据成员个数了?也不是

    • 这个结构体是多大?是24字节吗?

      //例 2
      struct str3
      {
      double a;
      int b;
      char c;
      };
      //初始化代码
      str3 str_3 = { 10.5,1024,'A' };
    • 答案是16字节,可以和上面的例子对比一下。

      这里写图片描述

    第一行8字节为str_3.a的内存
    第二行前4字节为str_3.b的内存
    第二行后4字节为str_3.c的内存

  • 总结一下
    可以把这种对齐方式比作使用特定大小的瓶子(箱子)装东西。如果将64位数据作为最大瓶子,那么装东西时所占的空间是以瓶子为单位的,你在大瓶子中装了一个小东西,那么它所占的空间还是大瓶子的空间(第一个例子的解释)。同时,大瓶子中还可以装小瓶子,小瓶子中还可以装更小的瓶子(第二个里的解释)

    编译器默认有内存优化,在编程是也可以手动修改对齐粒度
    #pragma pack(1) // 按照1字节方式进行对齐,括号中可以指定任意数
    #pragma pack() // 取消指定的对齐方式

二、存储顺序
  • 跟数组存储顺序一样,越往前的数据越在低地址处。

这篇关于C++ 结构体内存方式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1068061

相关文章

使用shardingsphere实现mysql数据库分片方式

《使用shardingsphere实现mysql数据库分片方式》本文介绍如何使用ShardingSphere-JDBC在SpringBoot中实现MySQL水平分库,涵盖分片策略、路由算法及零侵入配置... 目录一、ShardingSphere 简介1.1 对比1.2 核心概念1.3 Sharding-Sp

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

Spring创建Bean的八种主要方式详解

《Spring创建Bean的八种主要方式详解》Spring(尤其是SpringBoot)提供了多种方式来让容器创建和管理Bean,@Component、@Configuration+@Bean、@En... 目录引言一、Spring 创建 Bean 的 8 种主要方式1. @Component 及其衍生注解

python中的显式声明类型参数使用方式

《python中的显式声明类型参数使用方式》文章探讨了Python3.10+版本中类型注解的使用,指出FastAPI官方示例强调显式声明参数类型,通过|操作符替代Union/Optional,可提升代... 目录背景python函数显式声明的类型汇总基本类型集合类型Optional and Union(py

Linux系统管理与进程任务管理方式

《Linux系统管理与进程任务管理方式》本文系统讲解Linux管理核心技能,涵盖引导流程、服务控制(Systemd与GRUB2)、进程管理(前台/后台运行、工具使用)、计划任务(at/cron)及常用... 目录引言一、linux系统引导过程与服务控制1.1 系统引导的五个关键阶段1.2 GRUB2的进化优

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

C++ vector越界问题的完整解决方案

《C++vector越界问题的完整解决方案》在C++开发中,std::vector作为最常用的动态数组容器,其便捷性与性能优势使其成为处理可变长度数据的首选,然而,数组越界访问始终是威胁程序稳定性的... 目录引言一、vector越界的底层原理与危害1.1 越界访问的本质原因1.2 越界访问的实际危害二、基

IDEA与MyEclipse代码量统计方式

《IDEA与MyEclipse代码量统计方式》文章介绍在项目中不安装第三方工具统计代码行数的方法,分别说明MyEclipse通过正则搜索(排除空行和注释)及IDEA使用Statistic插件或调整搜索... 目录项目场景MyEclipse代码量统计IDEA代码量统计总结项目场景在项目中,有时候我们需要统计

C#和Unity中的中介者模式使用方式

《C#和Unity中的中介者模式使用方式》中介者模式通过中介者封装对象交互,降低耦合度,集中控制逻辑,适用于复杂系统组件交互场景,C#中可用事件、委托或MediatR实现,提升可维护性与灵活性... 目录C#中的中介者模式详解一、中介者模式的基本概念1. 定义2. 组成要素3. 模式结构二、中介者模式的特点

详解Java中三种状态机实现方式来优雅消灭 if-else 嵌套

《详解Java中三种状态机实现方式来优雅消灭if-else嵌套》这篇文章主要为大家详细介绍了Java中三种状态机实现方式从而优雅消灭if-else嵌套,文中的示例代码讲解详细,感兴趣的小伙伴可以跟... 目录1. 前言2. 复现传统if-else实现的业务场景问题3. 用状态机模式改造3.1 定义状态接口3