概述:
对于结构体:
字节对齐准则:(1),结构体变量的首地址能够被其最宽基本类型成员大小所整除;(2),结构体每个成员相对于结构体首地址的偏移量(offset)都是该成员大小的整数倍,如有需要,编译器会在成员之间加上中间填充字节;(3),结构体总大小为结构体最宽基本类型成员大小的整数倍,如有需要,编译器会在最末一个成员之后加上末尾填充字节。第(2)、(3)条准则决定了结构体变量占据内存空间的大小。
知识扩展1:若出现结构体嵌套,则准则(2)、(3)应改为:(2),复合成员相对于结构体首地址的偏移量是复合成员中最宽基本类型成员大小的整数倍;(3),结构体总大小为结构体最宽基本类型成员大小整数倍。
知识扩展2,若结构体内部出现数组,则准则2应为:(2),数组相对于结构体首地址的offset是该数组成员大小的整数倍。对于共用体:
原则上,共用体大小取决于占据最多内存的成员的长度。但字节对齐准则(3)仍然成立。
详情请参照以下分析:
对于结构体:
首先理解字节对齐准则:(1),结构体变量的首地址能够被其最宽基本类型成员大小所整除。(2),结构体每个成员相对于结构体首地址的偏移量(offset)都是该成员大小的整数倍,如有需要,编译器会在成员之间加上中间填充字节。(3),结构体总大小为结构体最宽基本类型成员大小的整数倍,如有需要,编译器会在最末一个成员之后加上末尾填充字节。第(2)、(3)条准则决定了结构体变量占据内存空间的大小。
基本类型指:char,short,int,float,double。数据宽度即其sizeof的大小。
eg: struct {char c; short s; float f;}A; struct{char c; float f; short s;}B; 则sizeof(A)=?,sizeof(B)=?
解:A中,char占1字节,short占2,则short相对于结构体首地址的offset(偏移量)=1,为满足准则(2),必须添加中间填充字节1;
float占4字节,相对于结构体首地址offset=(1+1+2)(红色为填充部分)即4,满足准则(2);
此时结构体大小为(1+1+2+4)=8(红色为填充部分),是最宽基本类型float整数倍,满足准则(3)。
所以可得出:sizeof(A)=8。
对于B:B中,char占1,float占4,则float相对于结构体首地址的offset=1,为满足准则(2),必须添加中间填充字节3;
short占2,相对于结构体首地址offset=(1+3+4)(红色为填充部分)即8,是short所占字节的整数倍,满足准则(2)。
此时结构体大小为(1+3+4+2)=10(红色为填充部分),不是最宽基本类型float的整数倍,这时,末尾必须添加末尾填充字节2,即(1+3+4+2 +2)=10(红色为填充部分)。
所以最后总大小为12;
总结:分析时只要这样即可:对于A:1 1 2 4 (红色为填充部分)对于B:1 3 4 2 2(红色为填充部分)这样就可以直观的得到结构体大小。迁移eg:struct{float f; int I; short s;}C; 则sizeof(C)=?(答案:12)
知识扩展1:若出现结构体嵌套,则准则(2)、(3)应改为:
(2),复合成员相对于结构体首地址的偏移量是复合成员中 最宽基本类型成员大小的整数倍。(3),结构体总大小为结构体 最宽基本类型成员大小整数倍。
eg: struct { float f; char c; double d; struct { double d1; float f1; double d2; char c1; }A; }B; 则sizeof(B)=?
解:先计算A大小:8 4 4 8 1 7(红色为填充部分),得到为32;
所以对于B,排列:4 1 3 8 32(红色为填充部分);
此时成员d的offset=8,前面填充3字节,满足准则2;
A的offset=16,是A中最宽基本类型double所占字节数(8)的整数倍,满足扩展准则2;
此时B大小为48,是B中最宽基本类型double的整数倍,满足扩展准则3.
所以最后sizeof(B)=48;
知识扩展2,若结构体内部出现数组,则准则2应为:(2),数组相对于结构体首地址的offset是该数组成员大小的整数倍。
eg :struct {double d[2];char*c;}C; 则sizeof(C)=?
解:double大小为8字节,所以double[2]为16;而指针大小均为4,所以C可排列为:16 4 4 (红色为填充部分)得到sizeof(C)=24;eg:struct {char*c; double d[2];}L;则sizeof(L)=?
解:char* c的大小为4,offset为0;double d[2]的大小为16,offset为double长度8的倍数,这里即取8;结构体总大小也是double长度8的整数倍。所以L可排列为:4 4 16 (红色为填充部分)得到sizeof(L) = 24;
对于共用体:
原则上,共用体大小取决于占据最多内存的成员的长度。但字节对齐准则(3)仍然成立。eg:
union { char c; double d; }e; 则sizeof(e)=8; union U { int a[5]; //20个字节 char b;//1个字节 double c;//8个字节 }; 则sizeof(union U) = 24;