这是工程师面试后的实际经历……

这道经典、易错的关于C语言结构体内存对齐的题目,你真的会吗:

求32bit环境下以下结构体所占的字节数:typedef struct test_struct{ char a;   short b;      char c;      int d; char e;}test_struct;

请说出你的答案:

下面看一下实际测试情况:

1、测试代码:

/*********************************** * 公众号:嵌入式大杂烩***********************************/#include 

typedef struct test_struct{ char a;   short b;      char c;      int d; char e;}test_struct;

int main(void){ test_struct test_s;  

 printf("\n============================================\n"); printf("test_s addr   = %#.8x\n", &test_s); printf("test_s.a addr = %#.8x\n", &test_s.a); printf("test_s.b addr = %#.8x\n", &test_s.b); printf("test_s.c addr = %#.8x\n", &test_s.c); printf("test_s.d addr = %#.8x\n", &test_s.d); printf("test_s.e addr = %#.8x\n", &test_s.e); printf("sizeof(test_s) = %d\n", sizeof(test_s)); printf("============================================\n");

 return 0;}

2、运行结果

在32bit环境中,该结构体所占的字节数为16。答对了吗?

嘿嘿,做个小调查(方便以后选题):

运行结果打印输出了很多重要的信息,从结果往前分析思路应该很清晰了吧?

不清晰也没关系,下面我们一起来分析分析:

3、分析

在分析这个问题之前,我们先记住关于结构体内存对齐的三条原则:

(1)结构体变量的起始地址能够被其最宽的成员大小整除。

(2)结构体每个成员相对于起始地址的偏移能够被其自身大小整除,如果不能则在前一个成员后面补充字节

(3)结构体总体大小能够被最宽的成员的大小整除,如不能则在后面补充字节

分析这个问题我们就不考虑编译器可以指定对齐大小的情况了。在32bit环境中,一般默认的对齐大小是4。

下面我们根据这三条原则来分析,并得出如下示意图:

从这张图中我们应该可以很清晰地看出整个结构体变量的内存占用情况。

如果还看不明白的朋友可阅读下面的解释(有点啰嗦,已经看明白的就不用看了~):

从上例的结果中,我们结构体变量test_s的起始地址为0x0028ff30,能够被其最宽的成员(int类型的d成员,占4个字节)整除,符合第(1)条原则。

a成员的地址即为结构体变量的起始地址0x0028ff30,排在a后面的是short类型(两个字节)的b成员。

根据第(2)条规则,显然b的地址不能从0x0028ff31开始,则编译器会在b成员的前一个成员(a成员)后边补1个空白字节,即b的的地址为从0x0028ff32,符合规则(2)。

b成员占两个字节,两个字节之后的地址为0x0028ff34,而c成员为char类型(1字节),则根据规则(2),c成员会存放至地址0x0028ff34处。

c成员占1个字节,1个字节之后的地址为0x0028ff35,排在c后面的是int类型(4个字节)的d成员,显然不能满足规则(2)。

编译器会在d成员的前一个成员(c成员)后面进行字节填充,这里必须填充3个字节才能符合规则(2),此时d会存放至地址0x0028ff38处。

d成员占4个字节,4个字节之后的地址为0x0028ff3c。根据规则(2),e成员可从该地址开始存放。

此时a+空白字节+b+c+空白字节+d+e所占的字节总数为13个字节,而结构体最宽的成员(int类型的d成员)所占字节数为4字节。

显然不能满足规则(3),编译器会在e成员后面填充3个字节。即整个结构体变量test_s所占的总字节数为16字节。

4、实际应用

(1)用保留变量替代填充字节

实际应用中我们可以上面的结构体变量改为:

typedef struct test_struct{ char a;   char reserve0;    /* 保留成员 */ short b;      char c;      int d; char e; char reserve1[3]; /* 保留成员 */}test_struct;

我们已经知道了编译器会自动给我们的结构体变量填充一些空白字节,这些填充字节我们是看不到的,是隐性的。

在结构体变量占用相同内存的情况下,我们可以显性的表示出这些填充字节,即创建一些保留成员 。

这样当我们需要给这个结构体添加一些成员时,我们可以把保留的成员替换为实际的成员。这样在一定程度下有利于我们节省内存空间。

(2)调整结构体成员的位置

从上面的分析中我们知道编译器会根据我们结构体成员的排列来进行空白字节填充以达到对齐的效果。

那么我们自己进行手动对齐一些成员,那就可以节省一些空间了。比如把上面的我们的test_struct结构体成员的顺序改为:

typedef struct test_struct{ char a;   char c;  short b;          int d; char e;}test_struct;

则结构体变量test_s所占的字节数变为12字节,即:

即比原来的16字节省下了4个字节。

虽然这点优化对于一般的嵌入式应用来说可能没什么必要,但是万一某一天真的需要在某些资源极其受限的嵌入式设备中开发应用,这就是可以优化的一点。

最后

以上就是本次的实验分享。如有错误,欢迎指出!谢谢

这道结构体内存对齐的题目很经典、也很容易出错,是嵌入式C语言笔试、面试题中的高频题目,很有必要弄清楚。

本文授权转载自公众号“嵌入式大杂烩”,作者ZhengN

printf打印结构体_工程师:这道题80%初学者都没做对!你确定搞懂结构体内存对齐了?...相关推荐

  1. java 链表放置结构体_结构体和它在链表中的使用

    一.结构体 由不同类型的数据组合成一个整体,以便引用,这些组合在一个整体中的数据是互相联系的. 1.1如何声明结构体呢? struct 结构体名  //结构体名字用作结构体类型的标志 {成员列表}; ...

  2. c语言 结构体_颖儿教你学C语言结构体,全面讲解,让程序小白玩转结构体编程...

    C语言结构体详细教学开始 前面的教程中我们讲解了数组(Array),它是一组具有相同类型的数据的集合.但在实际的编程过程中,我们往往还需要一组类型不同的数据,例如对于学生信息登记表,姓名为字符串,学号 ...

  3. malloc 结构体_算法与数据结构——结构体变量

    首先,要学习数据结构,一般要先了解结构体变量的使用,那么该如何定义结构体变量呢?随我一起回忆一下吧.(不一样的音乐,不一样的体验)(1)直接定义结构体变量.struct {int a;        ...

  4. 语言结构体在内存的分布_结构体内存对齐,这篇文章给你彻底搞会!(干货收藏)...

    脚本之家 你与百万开发者在一起 公众号:C语言编程 作者:薛定谔的coding猫 一.内存对齐的原因  1.平台原因(移植原因):一些资料上是这样说的,"不是所有的硬件平台都能访问任意地址上 ...

  5. c#往结构体里面读数据_结构体内存对齐,这回给你彻底搞会!

    一.内存对齐的原因  1.平台原因(移植原因):一些资料上是这样说的,"不是所有的硬件平台都能访问任意地址上的任意数据:某些硬件平台只能在某些特定地址处取某些特定的数据,否则就会抛出硬件异常 ...

  6. 遍历结构体_三菱ST语言编程(3)——结构体变量

    上篇文章介绍了数组,是一组相同类型数据的列表,那么不同类型的数据能否组合到一起用一个标签表示呢?答案当然是可以的,而实现这个功能的就是结构体(struct). 建立结构体 在三菱结构化编程的界面中左侧 ...

  7. 结构体内存对齐,默认对齐数,结构体传参

    我们先来计算一下结构体内存的大小 现在我们计算一下stu1和stu2每个成员内存偏移是多少. 在介绍偏移量之前,我们先简单介绍一下offsetof(是一个宏),它是用来计算结构体成员相较于起始位置的偏 ...

  8. 结构体及结构体内存对齐讲解

    文章目录 1.结构体类型的声明 2.结构体变量的定义和初始化 3.结构体内存对齐 4.结构体传参 1.结构体类型的声明 结构的基础知识 结构是一些值的集合,这些值称为成员变量.结构的每个成员可以是不同 ...

  9. 细讲C语言结构体(结构体内存对齐你懂了吗?)

    结构体 结构体类型的声明 结构体自引用 结构体变量的定义和初始化 结构体内存对齐 结构体传参 结构体 结构体的声明 结构体是一些值集合的,里面可以包括char,int,double等等的各种类型构成的 ...

最新文章

  1. 数据结构[单链表]操作大全(初始化,创建,插入,查询,删除,长度,输出)c++代码实现
  2. 开发日记-20190327 关键词 intant run原理
  3. openssl 生成公私钥
  4. Apache Hook机制解析(上)——钩子机制的实现
  5. 在网页中插入百度地图(实例)
  6. 为什么敏捷开发在亚洲实行不了
  7. rust电器元件需要什么材料_云南自考大专报考需要什么材料?
  8. 从0基础学Python:装饰器及练习(基础)
  9. 获取当前scn号scn1_Checkpoint和SCN的解析
  10. 2021年中国微粉磨料市场趋势报告、技术动态创新及2027年市场预测
  11. Is It A Tree?(并查集)
  12. EViews简介、下载安装教程
  13. [AngularJS] 插件ui-grid使用说明
  14. (转)中国电信友华PT921、PT921G光猫设置路由,无线WIFI设置
  15. Tumblr 的架构演进过程
  16. html文件做屏保win10,win10系统把屏保设置为桌面壁纸的处理方法
  17. angular自定义指令 directive
  18. 炉石传说 历代无面斩杀宇宙龙术
  19. 关于js中的类式继承
  20. 电商小程序实战教程-商品详情页开发

热门文章

  1. 未检测到其他显示器_如何将 Surface 连接到电视、显示器或投影仪,我教你
  2. MogDB存储过程事务控制与异常块
  3. 这样做,免费从Oracle同步数据
  4. Hadoop 面试,来看这篇就够了
  5. 万字通俗讲解何为复杂度
  6. Interrupted Exception异常可能没你想的那么简单!
  7. Rust 算法排位记-选择排序图示与代码实现
  8. 【华为云技术分享】《跟唐老师学习云网络》— iptables - filter过滤功能
  9. 计算机科学申请文书,美国留学:看牛人怎么写申请计算机CS专业的文书
  10. php 禁止ajax 重复请求,js ajax请求防止重复提交