(C语言)结构体内存对齐

简介: 1.对齐规则2.示例例1 例2 例33.存在内存对齐的原因1.平台原因2.性能原因4.修改默认对齐数

1.对齐规则

(1)结构体的第一个成员放在偏移量(相较于结构体变量的起始位置)为0的位置


(2)从第二个成员起,之后的每个成员都要对齐到对齐数的整数倍的地址处


       对齐数:编译器默认的对齐数与该成员大小的较小值


       VS中默认为8


       Linux中无默认对齐数,对齐数就是成员自身大小


(3)结构体总大小为最大对齐数(所有成员对齐数的最大值)的整数倍


(4)如果出现嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍


(5)当出现数组,数组的对齐数为数组元素与默认对齐数的较小值


2.示例

(以下均在VS下实现)


例1

#include <stdio.h>structs1{
charc1;
inti;
charc2;
};
intmain()
{
printf("%d", sizeof(structs1));//12return0;
}


计算结果为12,我们结合内存分布图来计算


c1为char类型,占1个字节,第一个成员放在偏移量为0的位置,(char的对齐数为1)


i为int类型,占4个字节,因此从偏移量为4的位置开始存放


c2为char类型,占1个字节,1为c2的默认对齐数,因此在i的后面接着存放c2,放在偏移量为9的位置处


结构体的最大对齐数为4,结构体的总大小为最大对齐数的整数倍,此时已经使用了9个字节的空间,则结构体的大小为12

2.png


例2

#include <stdio.h>structs1{
charc1;
inti;
charc2;
};
structs2{
charc;
structs1s;
inti;
};
intmain()
{
printf("%d\n", sizeof(structs1));//12printf("%d\n", sizeof(structs2));//20return0;
}


计算结果为20


c为char类型,占1个字节,第一个成员放在偏移量为0的位置,(char的对齐数为1)


由例1计算得出s的大小为12,s为结构体类型,对齐到自己的最大对齐数的整数倍处,s中的3个成员分别为char、int、char类型,最大对齐数为4,因此s对齐到偏移量为4的地址空间处。


i为int类型,对齐数为4,因此对齐到偏移量为16的地址空间处。


此时已经使用20个字节的空间,结构体的最大对齐数为4,因此结构体的大小为20

3.png


例3

#include <stdio.h>structs3{
charc1;
intarr[2];
charc2;
};
intmain()
{
printf("%d\n", sizeof(structs3));//16return0;
}


c1为char类型,占1个字节,第一个成员放在偏移量为0的位置,(char的对齐数为1)


arr[2]为数组,当出现数组时,对齐数为元素大小与默认对齐数的最小值,arr[2]的元素为int类型,对齐数为4,因此对齐到偏移量为4的地址空间处


c2为char类型,占1个字节,1为c2的默认对齐数,因此在arr[2]的后面接着存放c2,放在偏移量为12的位置处


4.png



3.存在内存对齐的原因

1.平台原因

并不是所有的硬件平台都可以访问任意地址上的任意数据。某些硬件平台只能在某些地址处取得某些特定类型的数据,否则会抛出硬件异常。


2.性能原因

数据结构(尤其是栈)应该尽可能地在自然边界上对齐。


原因:为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访问


可以说,结构体的内存对齐是拿空间时间  


当在使用结构体时,为了满足对齐,同时节省空间,可将占用空间小的成员尽量集中在一起


4.修改默认对齐数

结构在对齐方式不合适时,可以自己修改默认对齐数


使用#pragma预处理指令,可以修改默认对齐数(通常设置为2的次方数)

#include <stdio.h>#pragma pack(4)//设置默认对齐数为4structs1{
//  成员大小  默认对齐数  对齐数charc;//  1    4  1doubled;//8    4  4inti;//   4    4  4};
#pragma pack()//取消设置的默认对齐数,还原为默认对齐数structs2{
//  成员大小  默认对齐数  对齐数charc;//  1    8  1doubled;//8    8  8inti;//   4    8  4};
intmain()
{
printf("%d %d", sizeof(structs1),sizeof(structs2));
return0;
}

struct s1

struct s2

6.png

目录
相关文章
|
网络协议 编译器 Linux
【C语言】结构体内存对齐:热门面试话题
【C语言】结构体内存对齐:热门面试话题
554 0
|
存储 安全 C语言
【C语言程序设计——选择结构程序设计】预测你的身高(头歌实践教学平台习题)【合集】
分支的语句,这可能不是预期的行为,这种现象被称为“case穿透”,在某些特定情况下可以利用这一特性来简化代码,但在大多数情况下,需要谨慎使用。编写一个程序,该程序需输入个人数据,进而预测其成年后的身高。根据提示,在右侧编辑器补充代码,计算并输出最终预测的身高。分支下的语句,提示用户输入无效。常量的值必须是唯一的,且在同一个。语句的作用至关重要,如果遗漏。开始你的任务吧,祝你成功!,程序将会继续执行下一个。常量都不匹配,就会执行。来确保程序的正确性。
528 10
|
小程序 C语言
【C语言程序设计——基础】顺序结构程序设计(头歌实践教学平台习题)【合集】
目录 任务描述 相关知识 编程要求 测试说明 我的通关代码: 测试结果: 任务描述 相关知识 编程编写一个程序,从键盘输入3个变量的值,例如a=5,b=6,c=7,然后将3个变量的值进行交换,使得a=6,b=7,c=5。面积=sqrt(s(s−a)(s−b)(s−c)),s=(a+b+c)/2。使用输入函数获取半径,格式指示符与数据类型一致,实验一下,不一致会如何。根据提示,在右侧编辑器补充代码,计算并输出圆的周长和面积。
401 10
|
存储 编译器 C语言
【C语言程序设计——选择结构程序设计】求一元二次方程的根(头歌实践教学平台习题)【合集】
本任务要求根据求根公式计算并输出一元二次方程的两个实根,精确到小数点后两位。若方程无实根,则输出提示信息。主要内容包括: - **任务描述**:使用求根公式计算一元二次方程的实根。 - **相关知识**:掌握 `sqrt()` 函数的基本使用方法,判断方程是否有实根。 - **编程要求**:根据输入的系数,计算并输出方程的根或提示无实根。 - **测试说明**:提供两组测试数据及预期输出,确保代码正确性。 - **通关代码**:包含完整的 C 语言代码示例,实现上述功能。 通过本任务,你将学会如何处理一元二次方程的求解问题,并熟悉 `sqrt()` 函数的使用。
336 5
|
存储 算法 安全
【C语言程序设计——选择结构程序设计】按从小到大排序三个数(头歌实践教学平台习题)【合集】
本任务要求从键盘输入三个数,并按从小到大的顺序排序后输出。主要内容包括: - **任务描述**:实现三个数的排序并输出。 - **编程要求**:根据提示在编辑器中补充代码。 - **相关知识**: - 选择结构(if、if-else、switch) - 主要语句类型(条件语句) - 比较操作与交换操作 - **测试说明**:提供两组测试数据及预期输出。 - **通关代码**:完整代码示例。 - **测试结果**:展示测试通过的结果。 通过本任务,你将掌握基本的选择结构和排序算法的应用。祝你成功!
295 4
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
804 16
|
存储 算法 安全
【C语言程序设计——选择结构程序设计】求阶跃函数的值(头歌实践教学平台习题)【合集】
本任务要求输入x的值,计算并输出特定阶跃函数的结果。主要内容包括: 1. **选择结构基本概念**:介绍if、if-else、switch语句。 2. **主要语句类型**:详细解释if、if-else、switch语句的使用方法。 3. **跃迁函数中变量的取值范围**:说明如何根据条件判断变量范围。 4. **计算阶跃函数的值**:通过示例展示如何根据给定条件计算函数值。 编程要求:在右侧编辑器Begin-End之间补充代码,实现阶跃函数的计算和输出。测试说明提供了多个输入及其预期输出,确保代码正确性。最后提供通关代码和测试结果,帮助理解整个过程。
346 0
|
存储 算法 安全
【C语言程序设计——选择结构程序设计】判断一个数是不是5和7的倍数(头歌实践教学平台习题)【合集】
本任务要求输入一个正整数,判断其是否同时是5和7的倍数,若是输出&quot;Yes&quot;,否则输出&quot;No&quot;。内容涵盖选择结构的基本概念、主要语句类型(if、if-else、switch)及条件判断逻辑,帮助理解编程中的分支执行与条件表达式。测试用例包括正数、负数及非倍数情况,确保代码逻辑严谨。通关代码示例如下: ```cpp #include &quot;stdio.h&quot; int main(){ int a; scanf(&quot;%d&quot;, &a); if (a &lt;= 0){ printf(&quo
682 0
|
编译器 C语言 C++
【C语言程序设计——选择结构程序设计】求输入的日期是该年的第几天(头歌实践教学平台习题)【合集】
本任务要求编写程序,根据用户输入的年月日(以空格或回车分隔),计算并输出该天是该年的第几天,需注意判断闰年。主要内容包括: 1. **任务描述**:实现从键盘输入年月日,计算该天是当年的第几天。 2. **相关知识**: - `switch` 结构的基本语法及使用注意事项。 - 判断闰年的条件:能被4整除但不能被100整除,或能被400整除的年份为闰年。 3. **编程要求**:根据提示补充代码,确保程序正确处理输入并输出结果。 4. **测试说 示例代码展示了如何使用 `switch` 语句和闰年判断逻辑来完成任务。通过此练习,掌握 `switch` 语句的应用及闰年判断方法。
626 0
|
编译器 C语言 Python
C语言结构
C语言结构
133 0