呼和浩特资产评估公司,seo招聘网,清远医院网站建设方案,wordpress采集文章发布一、什么是对齐#xff0c;以及为什么要对齐#xff1a; 1. 现代计算机中内存空间都是按照byte划分的#xff0c;从理论上讲似乎对任何类型的变量的访问可以从任何地址开始#xff0c;但实际情况是在访问特定变量的时候经常在特定的内存地址访问#xff0c;这就需要各类型… 一、什么是对齐以及为什么要对齐 1. 现代计算机中内存空间都是按照byte划分的从理论上讲似乎对任何类型的变量的访问可以从任何地址开始但实际情况是在访问特定变量的时候经常在特定的内存地址访问这就需要各类型数据按照一定的规则在空间上排列而不是顺序的一个接一个的排放这就是对齐。 2. 对齐的作用和原因各个硬件平台对存储空间的处理上有很大的不同。一些平台对某些特定类型的数据只能从某些特定地址开始存取。其他平台可能没有这种情况 但是最常见的是如果不按照适合其平台的要求对数据存放进行对齐会在存取效率上带来损失。比如有些平台每次读都是从偶地址开始如果一个int型假设为 32位如果存放在偶地址开始的地方那么一个读周期就可以读出而如果存放在奇地址开始的地方就可能会需要2个读周期并对两次读出的结果的高低 字节进行拼凑才能得到该int数据。显然在读取效率上下降很多。这也是空间和时间的博弈。 二、对齐的实现 通常我们写程序的时候不需要考虑对齐问题。编译器会替我们选择适合目标平台的对齐策略。当然我们也可以通知给编译器传递预编译指令而改变对指定数据的对齐方法。 但是正因为我们一般不需要关心这个问题所以因为编辑器对数据存放做了对齐而我们不了解的话常常会对一些问题感到迷惑。最常见的就是struct数据结构的sizeof结果出乎意料。为此我们需要对对齐算法所了解。 对齐的算法 由于各个平台和编译器的不同现以本人使用的gcc version 3.2.2编译器32位x86平台为例子来讨论编译器对struct数据结构中的各成员如何进行对齐的。 设结构体如下定义 struct A { int a; char b; short c; }; 结构体A中包含了4字节长度的int一个1字节长度的char一个和2字节长度的short型数据一个。所以A用到的空间应该是7字节。但是因为编译器要对数据成员在空间上进行对齐。 所以使用sizeof(strcut A)值为8。 现在把该结构体调整成员变量的顺序。 struct B { char b; int a; short c; }; 这时候同样是总共7个字节的变量但是sizeof(struct B)的值却是12。 下面我们使用预编译指令#pragma pack (value)来告诉编译器使用我们指定的对齐值来取代缺省的。 #progma pack (2) /*指定按2字节对齐*/ struct C { char b; int a; short c; }; #progma pack () /*取消指定对齐恢复缺省对齐*/ sizeof(struct C)值是8。 修改对齐值为1 #progma pack (1) /*指定按1字节对齐*/ struct D { char b; int a; short c; }; #progma pack () /*取消指定对齐恢复缺省对齐*/ sizeof(struct D)值为7。 对于char型数据其自身对齐值为1对于short型为2对于int,float,double类型其自身对齐值为4单位字节。 这里面有四个概念值 1)数据类型自身的对齐值就是上面交代的基本数据类型的自身对齐值。 2)指定对齐值#pragma pack (value)时的指定对齐值value。 3)结构体或者类的自身对齐值其成员中自身对齐值最大的那个值。 4)数据成员、结构体和类的有效对齐值自身对齐值和指定对齐值中较小的那个值。 有了这些值我们就可以很方便的来讨论具体数据结构的成员和其自身的对齐方式。有效对齐值N是最终用来决定数据存放地址方式的值最重要。有效对齐N就是表示“对齐在N上”也就是说该数据的存放起始地址%N0.而数据结构中的数据变量都是按定义的先后顺序来排放的。第一个数据变量的起始地址就是 数据结构的起始地址。结构体的成员变量要对齐排放结构体本身也要根据自身的有效对齐值圆整(就是结构体成员变量占用总长度需要是对结构体有效对齐值的整 数倍结合下面例子理解)。这样就不难理解上面的几个例子的值了。 例子分析 分析例子B struct B { char b; int a; short c; }; 假设B从地址空间0x0000开始排放。该例子中没有定义指定对齐值在笔者环境下该值默认为4。第一个成员变量b的自身对齐值是1比指定或者默认指 定对齐值4小所以其有效对齐值为1所以其存放地址0x0000符合0x0000%10.第二个成员变量a其自身对齐值为4所以有效对齐值也为 4所以只能存放在起始地址为0x0004到0x0007这四个连续的字节空间中复核0x0004%40,且紧靠第一个变量。第三个变量c,自身对齐 值为2所以有效对齐值也是2可以存放在0x0008到0x0009这两个字节空间中符合0x0008%20。所以从0x0000到0x0009存 放的都是B内容。再看数据结构B的自身对齐值为其变量中最大对齐值(这里是b所以就是4所以结构体的有效对齐值也是4。根据结构体圆整的要求 0x0009到0x000010字节10240。所以0x0000A到0x000B也为结构体B所占用。故B从0x0000到0x000B 共有12个字节,sizeof(struct B)12; 同理,分析上面例子C #pragma pack (2) /*指定按2字节对齐*/ struct C { char b; int a; short c; }; #pragma pack () /*取消指定对齐恢复缺省对齐*/ 第一个变量b的自身对齐值为1指定对齐值为2所以其有效对齐值为1假设C从0x0000开始那么b存放在0x0000符合0x0000%1 0;第二个变量自身对齐值为4指定对齐值为2所以有效对齐值为2所以顺序存放在0x0002、0x0003、0x0004、0x0005四个连续 字节中符合0x0002%20。第三个变量c的自身对齐值为2所以有效对齐值为2顺序存放 在0x0006、0x0007中符合0x0006%20。所以从0x0000到0x00007共八字节存放的是C的变量。又C的自身对齐值为4所以 C的有效对齐值为2。又8%20,C只占用0x0000到0x0007的八个字节。所以sizeof(struct C)8. 有 了以上的解释相信你对C语言的字节对齐概念应该有了清楚的认识了吧。在网络程序中掌握这个概念可是很重要的喔在不同平台之间比如在Windows 和Linux之间传递2进制流比如结构体那么在这两个平台间必须要定义相同的对齐方式不然莫名其妙的出了一些错可是很难排查的哦^_^。
#includestdio.hint main()
{struct YourStruct { char dda; double dda1; int type; }; #pragma pack(push) //保存对齐状态 #pragma pack(4) //设定为4字节对齐 struct MyStruct { char dda; double dda1; int type; }; printf(%d\n,sizeof(struct MyStruct));#pragma pack(pop) //恢复对齐状态printf(%d\n,sizeof(struct YourStruct));return 0;
} C对象内存布局
http://www.cnblogs.com/lindeshi/archive/2012/10/20/2732590.html
http://www.cnblogs.com/lindeshi/archive/2012/10/20/2732592.html 查看虚函数表
http://www.cnblogs.com/mhjerry/archive/2012/12/16/2820895.html很不错的一篇文章