`
tyytzy
  • 浏览: 38156 次
  • 性别: Icon_minigender_1
  • 来自: 安徽
社区版块
存档分类
最新评论

C语言中的内存对齐问题

 
阅读更多

.C语言中的内存对齐问题! 2011-9-26 21:43阅读(20).赞赞赞赞转载(2)分享评论复制地址举报更多上一篇 |下一篇:《让人吐血的谜语... 

 

当在C中定义了一个结构类型时,它的大小是否等于各字段(field)大小之和?编译器将如何在内存中放置这些字段?ANSI C对结构体的内存布局有什么要求?而我们的程序又能否依赖这种布局?这些问题或许对不少朋友来说还有点模糊,那么本文就试着探究它们背后的秘密。

 

    首先,至少有一点可以肯定,那就是ANSI C保证结构体中各字段在内存中出现的位置是随它们的声明顺序依次递增的,并且第一个字段的首地址等于整个结构体实例的首地址。比如有这样一个结构体:

 

  struct vector{int x,y,z;} s;

  int *p,*q,*r;

  struct vector *ps;

 

  p = &s.x;

  q = &s.y;

  r = &s.z;

  ps = &s;

 

  assert(p < q);

  assert(p < r);

  assert(q < r);

  assert((int*)ps == p);

  // 上述断言一定不会失败

 

    这时,有朋友可能会问:"标准是否规定相邻字段在内存中也相邻?"。 唔,对不起,ANSI C没有做出保证,你的程序在任何时候都不应该依赖这个假设。那这是否意味着我们永远无法勾勒出一幅更清晰更精确的结构体内存布局图?哦,当然不是。不过先让我们从这个问题中暂时抽身,关注一下另一个重要问题————内存对齐。

 

    许多实际的计算机系统对基本类型数据在内存中存放的位置有限制,它们会要求这些数据的首地址的值是某个数k(通常它为4或8)的倍数,这就是所谓的内存对齐,而这个k则被称为该数据类型的对齐模数(alignment modulus)。当一种类型S的对齐模数与另一种类型T的对齐模数的比值是大于1的整数,我们就称类型S的对齐要求比T强(严格),而称T比S弱(宽松)。这种强制的要求一来简化了处理器与内存之间传输系统的设计,二来可以提升读取数据的速度。比如这么一种处理器,它每次读写内存的时候都从某个8倍数的地址开始,一次读出或写入8个字节的数据,假如软件能保证double类型的数据都从8倍数地址开始,那么读或写一个double类型数据就只需要一次内存操作。否则,我们就可能需要两次内存操作才能完成这个动作,因为数据或许恰好横跨在两个符合对齐要求的8字节内存块上。某些处理器在数据不满足对齐要求的情况下可能会出错,但是Intel的IA32架构的处理器则不管数据是否对齐都能正确工作。不过Intel奉劝大家,如果想提升性能,那么所有的程序数据都应该尽可能地对齐。Win32平台下的微软C编译器(cl.exe for 80x86)在默认情况下采用如下的对齐规则: 任何基本数据类型T的对齐模数就是T的大小,即sizeof(T)。比如对于double类型(8字节),就要求该类型数据的地址总是8的倍数,而char类型数据(1字节)则可以从任何一个地址开始。Linux下的GCC奉行的是另外一套规则(在资料中查得,并未验证,如错误请指正):任何2字节大小(包括单字节吗?)的数据类型(比如short)的对齐模数是2,而其它所有超过2字节的数据类型(比如long,double)都以4为对齐模数。

 

    现在回到我们关心的struct上来。ANSI C规定一种结构类型的大小是它所有字段的大小以及字段之间或字段尾部的填充区大小之和。嗯?填充区?对,这就是为了使结构体字段满足内存对齐要求而额外分配给结构体的空间。那么结构体本身有什么对齐要求吗?有的,ANSI C标准规定结构体类型的对齐要求不能比它所有字段中要求最严格的那个宽松,可以更严格(但此非强制要求,VC7.1就仅仅是让它们一样严格)。我们来看一个例子(以下所有试验的环境是Intel Celeron 2.4G + WIN2000 PRO + vc7.1,内存对齐编译选项是"默认",即不指定/Zp与/pack选项):

 

  typedef struct ms1

  {

     char a;

     int b;

  } MS1;

 

    假设MS1按如下方式内存布局(本文所有示意图中的内存地址从左至右递增):

       _____________________________

       |        |                                              |

       |   a   |        b                                    |

       |        |                                              |

       +------------------------- - ---------------+

 Bytes:    1             4

 

    因为MS1中有最强对齐要求的是b字段(int),所以根据编译器的对齐规则以及ANSI C标准,MS1对象的首地址一定是4(int类型的对齐模数)的倍数。那么上述内存布局中的b字段能满足int类型的对齐要求吗?嗯,当然不能。如果你是编译器,你会如何巧妙安排来满足CPU的癖好呢?呵呵,经过1毫秒的艰苦思考,你一定得出了如下的方案:

 

       _______________________________________

       |       |\\\\\\\\\\\|                 |

       |   a   |\\padding\\|       b         |

       |       |\\\\\\\\\\\|                 |

       +-------------------------------------+

 Bytes:    1         3             4

 

    这个方案在a与b之间多分配了3个填充(padding)字节,这样当整个struct对象首地址满足4字节的对齐要求时,b字段也一定能满足int型的4字节对齐规定。那么sizeof(MS1)显然就应该是8,而b字段相对于结构体首地址的偏移就是4。非常好理解,对吗?现在我们把MS1中的字段交换一下顺序:

 

  typedef struct ms2

  {

     int a;

     char b;

  } MS2;

 

    或许你认为MS2比MS1的情况要简单,它的布局应该就是

 

       _______________________

       |              |                             |

       |     a       |   b                        |

       |              |                             |

       +------------ - --------------------+

 Bytes:      4           1 

 

    因为MS2对象同样要满足4字节对齐规定,而此时a的地址与结构体的首地址相等,所以它一定也是4字节对齐。嗯,分析得有道理,可是却不全面。让我们来考虑一下定义一个MS2类型的数组会出现什么问题。C标准保证,任何类型(包括自定义结构类型)的数组所占空间的大小一定等于一个单独的该类型数据的大小乘以数组元素的个数。换句话说,数组各元素之间不会有空隙。按照上面的方案,一个MS2数组array的布局就是:

 

|<-    array[1]     ->|<-    array[2]     ->|<- array[3] .....

 

__________________________________________________________

|              |        |               |       |

|     a       |   b   |      a       |   b  |.............

|              |        |               |       |

+----------------------------------------------------------

Bytes:  4         1          4           1

 

    当数组首地址是4字节对齐时,array[1].a也是4字节对齐,可是array[2].a呢?array[3].a ....呢?可见这种方案在定义结构体数组时无法让数组中所有元素的字段都满足对齐规定,必须修改成如下形式:

 

       ___________________________________

       |              |        |\\\\\\\\ \ \\\ |

       |     a       |   b   |\\padding\\|

       |              |        |\\\\\\\\\\ \ \\|

       +---------------------------------+

 Bytes:      4           1         3

 

    现在无论是定义一个单独的MS2变量还是MS2数组,均能保证所有元素的所有字段都满足对齐规定。那么sizeof(MS2)仍然是8,而a的偏移为0,b的偏移是4。

 

    好的,现在你已经掌握了结构体内存布局的基本准则,尝试分析一个稍微复杂点的类型吧。

 

  typedef struct ms3

  {

     char a;

     short b;

     double c;

  } MS3;

 

    我想你一定能得出如下正确的布局图:

 

        padding  

           |

      _____v_________________________________

      |    |\|      |\\\\\\\\\|               |

      | a |\|  b  |\padding\|       c   |

      |    |\|      |\\\\\\\\\|               |

      +-------------------------------------+

Bytes:  1  1   2       4            8

 

    sizeof(short)等于2,b字段应从偶数地址开始,所以a的后面填充一个字节,而sizeof(double)等于8,c字段要从8倍数地址开始,前面的a、b字段加上填充字节已经有4 bytes,所以b后面再填充4个字节就可以保证c字段的对齐要求了。sizeof(MS3)等于16,b的偏移是2,c的偏移是8。接着看看结构体中字段还是结构类型的情况:

 

  typedef struct ms4

  {

     char a;

     MS3 b;

  } MS4;

 

    MS3中内存要求最严格的字段是c,那么MS3类型数据的对齐模数就与double的一致(为8),a字段后面应填充7个字节,因此MS4的布局应该是:

       _______________________________________

       |       |\\\\\\\\\\\|                 |

       |   a   |\\padding\\|       b         |

       |       |\\\\\\\\\\\|                 |

       +-------------------------------------+

 Bytes:    1         7             16

 

    显然,sizeof(MS4)等于24,b的偏移等于8。

 

    在实际开发中,我们可以通过指定/Zp编译选项来更改编译器的对齐规则。比如指定/Zpn(VC7.1中n可以是1、2、4、8、16)就是告诉编译器最大对齐模数是n。在这种情况下,所有小于等于n字节的基本数据类型的对齐规则与默认的一样,但是大于n个字节的数据类型的对齐模数被限制为n。事实上,VC7.1的默认对齐选项就相当于/Zp8。仔细看看MSDN对这个选项的描述,会发现它郑重告诫了程序员不要在MIPS和Alpha平台上用/Zp1和/Zp2选项,也不要在16位平台上指定/Zp4和/Zp8(想想为什么?)。改变编译器的对齐选项,对照程序运行结果重新分析上面4种结构体的内存布局将是一个很好的复习。

 

    到了这里,我们可以回答本文提出的最后一个问题了。结构体的内存布局依赖于CPU、操作系统、编译器及编译时的对齐选项,而你的程序可能需要运行在多种平台上,你的源代码可能要被不同的人用不同的编译器编译(试想你为别人提供一个开放源码的库),那么除非绝对必需,否则你的程序永远也不要依赖这些诡异的内存布局。顺便说一下,如果一个程序中的两个模块是用不同的对齐选项分别编译的,那么它很可能会产生一些非常微妙的错误。如果你的程序确实有很难理解的行为,不防仔细检查一下各个模块的编译选项。

 

    思考题:请分析下面几种结构体在你的平台上的内存布局,并试着寻找一种合理安排字段声明顺序的方法以尽量节省内存空间。

 

    A. struct P1 { int a; char b; int c; char d; };

    B. struct P2 { int a; char b; char c; int d; };

    C. struct P3 { short a[3]; char b[3]; };

    D. struct P4 { short a[3]; char *b[3]; };

    E. struct P5 { struct P2 *a; char b; struct P1 a[2];  };

 

参考资料:

 

    【1】《深入理解计算机系统(修订版)》,

         (著)Randal E.Bryant; David O'Hallaron,

         (译)龚奕利 雷迎春,

         中国电力出版社,2004

 

    【2】《C: A Reference Manual》(影印版),

         (著)Samuel P.Harbison; Guy L.Steele,

         人民邮电出版社,2003

 

分享到:
评论

相关推荐

    C语言结构体内存对齐问题浅析.pdf

    C语言结构体内存对齐问题浅析

    C语言结构体内存对齐问题.pdf

    在C语言中,内存对齐指的是编译器在处理结构体或联合体时,出于优化CPU读取内存的效率,按照一定的规则调整结构体成员变量的存储地址。每个不同的硬件平台都有其推荐的对齐方式,最常见的是4字节对齐(32位系统)和8...

    C语言内存对齐,提高寻址效率

    了解和掌握C语言内存对齐的机制对于编写高效且性能优越的程序至关重要。特别是对于初学指针和系统底层操作的C语言学生而言,深入理解内存对齐不仅可以帮助他们写出运行更快的代码,还能帮助他们更好地理解计算机底层...

    C语言数据存储对齐详解

    C语言中的数据存储对齐是编译器为了提高内存访问效率和硬件兼容性而采用的一种策略。它涉及到如何在内存中安排数据结构的各个成员,确保数据读取和写入时能够快速高效地进行。对齐规则主要有以下几点: 1. **成员按...

    C语言中的内存对齐:原理、实践与性能优化

    内存对齐是C语言中一个重要的概念,它影响着数据在内存中的存储方式和访问效率。通过深入理解内存对齐的原理和应用,程序员可以写出更高效、更可靠的代码,提高程序的性能和稳定性。在实际编程中,合理地利用内存...

    C语言字节对齐问题详解

    C语言中的字节对齐问题是指在编译器将C语言程序编译成机器代码时,对变量的存储方式进行调整,以提高存取效率和减少存储空间。字节对齐是指在内存中将变量按照一定的规则排列,以便在访问变量时可以提高存取效率。 ...

    C语言内存对齐.pdf

    在计算机编程中,内存对齐是一种重要的优化策略,特别是在C语言中。它涉及到数据结构在内存中的布局方式,以确保高效访问和存储。内存对齐的主要目的是提高数据存取速度,减少处理器的等待时间,并充分利用硬件的...

    C语言实现内存管理

    - **内存对齐**:C编译器通常会进行内存对齐,确保变量在内存中的位置符合特定规则,提高数据访问效率。 - **栈与堆**:栈内存由编译器自动管理,用于存放局部变量和函数调用信息;堆内存需要程序员手动管理,适用...

    C语言内存精讲,让你彻底明白C语言的运行机制!

    4. C语言内存对齐,提高寻址效率 5. 内存分页机制,完成虚拟地址的映射 6. 分页机制究竟是如何实现的? 7. MMU部件以及对内存权限的控制 8. Linux下C语言程序的内存布局(内存模型) 9. Windows下C语言程序的内存...

    C语言字节对齐详解

    在编写C语言程序时,通常不需要手动管理对齐的问题。编译器会自动选择适合目标平台的对齐策略。然而,有时我们可能需要自己干预对齐的方式,尤其是在处理结构体时。接下来,我们将通过具体的示例来探讨编译器是如何...

    C语言内存精讲.rar

    4. **内存对齐**:C语言中的内存对齐是为了优化数据访问效率,确保数据在内存中的位置符合特定的规则。了解并掌握对齐规则,可以避免不必要的性能损失。 5. **指针与内存**:指针是C语言中强大的工具,它可以直接...

    c语言中动态内存分配问题

    在C语言中,动态内存分配是一项重要的编程技巧,它允许程序员在程序运行时根据需要分配内存,而不是在编译时预设固定的内存空间。...在实际编程中,处理动态内存时,还要考虑到内存碎片和内存对齐等问题。

    语言:内存字节对齐详解

    理解内存对齐至关重要,因为它直接影响到程序的性能和跨平台兼容性。 首先,我们要明白什么是内存对齐。在计算机系统中,内存是以字节为基本单位进行分配的,但并不是所有数据都可以从任意地址开始存储。为了优化...

    C语言实现内存池

    - **内存对齐**:为了提高内存访问效率,内存池可能需要支持内存对齐,确保分配的内存地址满足特定的对齐要求。 - **多线程支持**:在多线程环境下,内存池需要提供线程安全的分配和释放操作,可能需要使用锁或者...

    5分钟搞定内存字节对齐

    C语言内存字节对齐详解 在C语言中,内存字节对齐是指编译器为了提高程序执行效率和可移植性,而对结构体成员在内存中的存储方式进行的调整。这个调整是基于体系结构的对齐规则,旨在提高程序的执行效率和可移植性。...

    内存对齐 详细介绍内存对齐的原理

    以C语言为例,大多数编译器默认会对结构体中的成员进行内存对齐,以提高程序的运行效率。 **示例代码分析**: ```c struct foo { char c1; short s; char c2; int i; }; ``` 在这个例子中,`struct foo`包含了...

Global site tag (gtag.js) - Google Analytics