一篇文章带你了解C语言内存对齐公式

 更新时间:2021年08月15日 10:57:59   作者:Wallace Zhang  
这篇文章主要介绍了C语言内存对齐,包括内存对其的基本概念及用法,以及注意事项,并以实例形式加以说明,需要的朋友可以参考下,希望能给你带来帮助

一、前言

每一个特定平台上的编译器都有自己的默认“对齐系数”(也叫对齐模数)。GCC中默认#program pack(4),即4个字节的内存对齐。Keil也是采用4字节对齐的。也可以通过预编译命令#pragma pack(n),n = 1,2,4,8,16来改变这一系数,一般情况下尽量使用自然对齐系数,不要修改它。

STM32单片机上各个变量占用的字节数:

在这里插入图片描述

在这里插入图片描述

二、公式

公式一、结构体变量里,成员的起始地址必须满足 : 起始地址 % 成员的字节数(sizeof值)= 0 (说白了就是能整除)

公式二、结构体变量的总字节数必须满足:总字节数 % 最大的成员字节数 = 0 (说白了就是能整除)

2.1、例子一

struct te_a{
	
  /* 公式一 */
	char a;   /* a的起始地址0x00,然后用公式一计算:0x00 % 1(char为1个字节) = 0,所以成员a占用了内存0x00   */
	int  b;   /* b的起始地址0x01 % 4(int为4个字节)不等于0,那么再计算0x02%4还是不等于0,直到0x04 % 4 = 0 ,所以成员b占用了内存0x04 ~ 0x07 */
	char c;   /* 成员b的结尾地址是0x07,所以成员c从0x08开始计算,那么计算0x08 % 1 = 0 , 所以成员c占用了内存0x08 */

}Test1;

OK,经过公式一的运算后,结构体里成员的分布如下:

在这里插入图片描述

经过公式一的计算后,结构体变量Test1的大小是9个字节。内存对齐的计算还没有结束,接着使用公式二计算:

结构体变量的总字节数 % 最大的成员字节数 = 0 , 在结构体变量Test1里,最大的成员是b,b的大小是4个字节。那么,当前的结构体变量大小9字节 % 4字节 等于 0 。当结构体变量大小为12字节 % 4字节 = 0,所以最终结构体变量Test1占用的内存字节数是12,其内存的分布如下:

在这里插入图片描述

以上的都是根据公式计算出来的结果,那实际在单片机里是不是这样呢?把代码下载到STM32单片机里,进入DEBUG模式看看。

在这里插入图片描述

在这里插入图片描述

从以下的内存分布看来,公式一与公式二的计算没有问题。

在这里插入图片描述

2.2、例子二

struct te_a{
	
  /* 公式一 */
	int   a;  /* a的起始地址是0x00,然后根据公式一计算0x00 % 4 = 0 ,那么成员a占用的内存是0x00 ~ 0x03 */
	float b;  /* b的起始地址是0x04, 然后根据公式一计算0x04 % 4 = 0 ,那么成员b占用的内存是0x04 ~ 0x07 */
	char  c;  /* c的起始地址是0x08, 然后根据公式一计算0x08 % 1 = 0 ,那么成员c占用的内存是0x08 */

}Test1;

OK,经过公式一的运算后,结构体里成员应该占用9个字节的内存,内存的分布如下:

在这里插入图片描述

接着根据公式二的运算,结构体的总字节数 % 最大的成员字节数 = 0, 可以轻松得出结构体的总字节数 = 12时,满足12 % 4 = 0。所以经过公式二的计算后,内存分布如下:

在这里插入图片描述

把代码烧录到STM32,进入Debug模式看看。

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

2.3、例子三

struct te_a{
	
  /* 公式一 */
	int     a;  /* a的起始地址是0x00,然后根据公式一计算0x00 % 4 = 0 ,那么成员a占用的内存是0x00 ~ 0x03 */
	float   b;  /* b的起始地址是0x04, 然后根据公式一计算0x04 % 4 = 0 ,那么成员b占用的内存是0x04 ~ 0x07 */
	double  c;  /* c的起始地址是0x08, 然后根据公式一计算0x08 % 8 = 0 ,那么成员c占用的内存是0x08 ~ 0x0F */

}Test1;

OK,经过公式一的运算后,结构体里成员应该占用16个字节的内存,内存的分布如下:

在这里插入图片描述

接着根据公式二的运算,结构体的总字节数 % 最大的成员字节数 = 0, 那么16 % 8 = 0,运气非常好,公式二不用补位就能让公式二成立。所以经过公式二的运算后,内存还是一样的:

在这里插入图片描述

把代码烧录到STM32,进入Debug模式看看。

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

总结

本篇文章就到这里了,希望能给你带来帮助,也希望您能够多多关注脚本之家的更多内容!

相关文章

  • C语言中的编码小技巧

    C语言中的编码小技巧

    这篇文章主要介绍了C语言中的编码小技巧,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-02-02
  • C实现分子沉积模拟的示例代码

    C实现分子沉积模拟的示例代码

    这篇文章主要介绍了计算机在材料科学中的一个练习题,功能是模拟气化后分子沉积
    2013-11-11
  • C++详解如何实现单链表

    C++详解如何实现单链表

    线性表的链式存储又称为单链表,它是指通过一组任意的存储单元来存储线性表中的数据元素。本文将用C++实现单链表,需要的可以参考一下
    2022-06-06
  • C++ Boost StringAlgorithms超详细讲解

    C++ Boost StringAlgorithms超详细讲解

    Boost是为C++语言标准库提供扩展的一些C++程序库的总称。Boost库是一个可移植、提供源代码的C++库,作为标准库的后备,是C++标准化进程的开发引擎之一,是为C++语言标准库提供扩展的一些C++程序库的总称
    2022-11-11
  • 解析C语言与C++的编译模型

    解析C语言与C++的编译模型

    C++继承了C的编译模型,C语言的编译链接模型相对简洁,但C++继承了这些机制之后变得更加复杂难以理解,这里就来带大家简要解析C语言与C++的编译模型
    2016-05-05
  • Qt中鼠标点击的几种状态

    Qt中鼠标点击的几种状态

    在Qt中,鼠标点击按钮通常会触发一系列的事件,包括pressed、released、clicked等,本文主要介绍了Qt中鼠标点击的几种状态,具有一定的参考价值,感兴趣的可以了解一下
    2023-12-12
  • C++编写DLL动态链接库的步骤与实现方法

    C++编写DLL动态链接库的步骤与实现方法

    这篇文章主要介绍了C++编写DLL动态链接库的步骤与实现方法,结合实例形式分析了C++导出类文件及生成与调用DLL动态连接库的相关操作技巧,需要的朋友可以参考下
    2016-08-08
  • C语言实现贪吃蛇小游戏

    C语言实现贪吃蛇小游戏

    这篇文章主要为大家详细介绍了C语言实现贪吃蛇小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03
  • C语言实现井字棋(三子棋)

    C语言实现井字棋(三子棋)

    这篇文章主要为大家详细介绍了C语言实现井字棋游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-04-04
  • C++超详细讲解友元的使用

    C++超详细讲解友元的使用

    采用类的机制后实现了数据的隐藏与封装,类的数据成员一般定义为私有成员,成员函数一般定义为公有的,依此提供类与外界间的通信接口。但是,有时需要定义一些函数,这些函数不是类的一部分,但又需要频繁地访问类的数据成员,这时可以将这些函数定义为该类的友元函数
    2022-04-04

最新评论