C++结构体(struct/typedef)

2024-06-13 12:58
文章标签 c++ 结构 struct typedef

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

定义结构体需要使用关键字struct修饰

struct MyStruct1
{	int a;int c;
};

(重要)下面声明了4个东西,分别是
1.结构体User
2.User指针类型的指针p1
3.User指针类型的指针p3
4.给User起了一个别名叫p2

// 注意p1p3是指针,p2不是
typedef struct User
{char name[6];int  age;
}* p1,p2,*p3;

使用指针偏移的方式对上面结构体赋值(重要)

User u;
p1 p_user = &u;
// ->推导符号表示指针偏移
p_user->age = 5;
p_user->name[0] = 'a';
// *号叫做间接运算符
std::cout << (*p_user).name[0] << std::endl;
std::cout << p_user->name[0] << std::endl;

结构体是一块连续的内存,下面我将定义2个结构体,并打印出他们的大小

struct MyStruct1
{char  a;double b;int c;
};
int main()
{MyStruct1 mys1;// 打印出24std::cout << sizeof(mys1) << "\n";
}

因为结构体的字节对齐特性导致结构体的大小不一样,下面是我自己对字节对齐特性的理解,仅供同行参考,因为每个人有每个人自己的学习方法与理解方式
字节对齐
我定义了一个结构体,如下,因为为了阐述字节对齐这个特性,我觉得上面两个结构体有些复杂,所以下面的结构体用来解释字节对齐

struct MyStruct3
{char  a;//1个字节int c;//4个字节short b;//2个字节
};

MyStruct3在内存中实际是这样的,其中1表示内存中第1个字节,2表示内存中第2个字节,以此类推,12表示内存中的第12个字节
在这里插入图片描述
从图中可以看出,
char a占用了黄色的内存,是4个字节
int c占用了绿色的内存,4个字节
short b占用了浅蓝的内存,4个字节
因为
int4个字节>short2个字节>char1个字节,所以该结构体所有类型都向int看齐
又因为
先定义的char,之后定义的int,之后定义的short
所以从图中可以看出第1排是char,第2排是int,第3排是short

综上两个"因为",所以才出现上述的图,它占用了12个字节

下面我又定义了一个结构体

struct MyStruct4
{int c;//4个字节char  a;//1个字节short b;//2个字节
};

MyStruct4在内存中是下面这样存在的
在这里插入图片描述
其中绿色依然表示int c,黄色表示char a,蓝色表示short b,从图中可知结构体MyStruct4占了8个字节

综上所述,我个人通常会把结构体的"字节对齐",理解成,“根据最长类型的字节补全内存”

但是一个新的问题产生了,为什么要出现字节对齐这种性质呢??
我们回到上面的MyStruct3

struct MyStruct3
{char  a;//1个字节,我依然用黄色表示int c;//4个字节,我依然用绿色表示short b;//2个字节,我依然用蓝色表示
};

现在我们进行几个假设:
假设1不存在字节对齐这个概念
那么MyStruct3在内存中的数据应该是下面这样子的,很简单,就是按照顺序排列呗
在这里插入图片描述
黄色表示char a,可以看出占用1个字节
蓝色表示short b,可以看出占用2个字节
绿色表示int c,可以看出占用4个字节
最后的内存8没有使用到,综上,假设不存在字节对齐这个概念,那么该结构体占用7个字节
接下来,我在假设1的基础上,进行假设2

假设2:CPU一次只读取一个字节
若CPU一次只读取一个字节,这是没有什么问题的,完全不需要字节对齐,因为无论如何,CPU如果一次只读取一个字节,那么某个数据占几个字节,CPU就要读取几次数据,所以,若CPU一次只能读取1个字节,那么就不需要,也没必要存在字节对齐

可是,现在的CPU一次并不是读取一个字节,现代CPU一次都是读取4字节,或者8字节,那么这就出现个问题,根据上面的图,可以看出若一次读取4字节,那么,会读取1234,然而4是int c中的第一个字节,也就是说,这会读取int 前四分之一,后四分之三没有读取到,若是再读int,只能读取后面四分之三,然后将前面四分之一和后面读取的四分之三拼接起来,所以我个人认为字节对齐就是为了避免拆分字(word)的一种行为,因为拆分字之后,再将字组合,会很麻烦

这篇关于C++结构体(struct/typedef)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

Redis中Set结构使用过程与原理说明

《Redis中Set结构使用过程与原理说明》本文解析了RedisSet数据结构,涵盖其基本操作(如添加、查找)、集合运算(交并差)、底层实现(intset与hashtable自动切换机制)、典型应用场... 目录开篇:从购物车到Redis Set一、Redis Set的基本操作1.1 编程常用命令1.2 集

C++读写word文档(.docx)DuckX库的使用详解

《C++读写word文档(.docx)DuckX库的使用详解》DuckX是C++库,用于创建/编辑.docx文件,支持读取文档、添加段落/片段、编辑表格,解决中文乱码需更改编码方案,进阶功能含文本替换... 目录一、基本用法1. 读取文档3. 添加段落4. 添加片段3. 编辑表格二、进阶用法1. 文本替换2

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别

C++右移运算符的一个小坑及解决

《C++右移运算符的一个小坑及解决》文章指出右移运算符处理负数时左侧补1导致死循环,与除法行为不同,强调需注意补码机制以正确统计二进制1的个数... 目录我遇到了这么一个www.chinasem.cn函数由此可以看到也很好理解总结我遇到了这么一个函数template<typename T>unsigned

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

Vite 打包目录结构自定义配置小结

《Vite打包目录结构自定义配置小结》在Vite工程开发中,默认打包后的dist目录资源常集中在asset目录下,不利于资源管理,本文基于Rollup配置原理,本文就来介绍一下通过Vite配置自定义... 目录一、实现原理二、具体配置步骤1. 基础配置文件2. 配置说明(1)js 资源分离(2)非 JS 资

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

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

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

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