#pragma pack(1)//结构体全部1字节对齐
typedef struct xxx
{
TCHARxyz;
longabc;
}X,*X;
#pragma pack()//取消自定义的对齐方式
例如:
#pragma pack(8)
struct s1{
short a;
long b;
};
struct s2{
char c;
s1 d;
long long e;
};
#pragma pack()
问 :
1.sizeof(s2) = ?
2.s2的c后面空了几个字节接着是d?
解答,结果如下:
sizeof(S2)结果为24.
成员对齐有一个重要的条件,即每个成员分别对齐.即每个成员按自己的方式对齐.
也就是说上面虽然指定了按8字节对齐,但并不是所有的成员都是以8字节对齐.
其对齐的规则是,每个成员按其类型的对齐参数(通常是这个类型的大小)和指定对齐参数(这里是8字节)中较小的一个对齐.
并且结构的长度必须为所用过的所有对齐参数的整数倍,不够就补空字节.
S1中,成员a是1字节默认按1字节对齐,指定对齐参数为8,这两个值中取1,a按1字节对齐;成员b是4个字节,默认是按4字节对齐,这时就按4字节对齐,所以sizeof(S1)应该为8;
S2中,c和S1中的a一样,按1字节对齐,而d 是个结构,它是8个字节,它按什么对齐呢?
对于结构来说,它的默认对齐方式就是它的所有成员使用的对齐参数中最大的一个,S1的就是4.
所以,成员d就是按4字节对齐.成员e是8个字节,它是默认按8字节对齐,和指定的一样,所以它对到8字节的边界上,这时,已经使用了12个字节了,所以又添加了4个字节的空,从第16个字节开始放置成员e.
这时,长度为24,已经可以被8(成员e按8字节对齐)整除.这样,一共使用了24个字节.
a b
S1的内存布局: 11**,1111,
c S1.a S1.b d
S2的内存布局: 1***,11**,1111,****11111111
这里有三点很重要:
1.每个成员分别按自己的方式对齐,并能最小化长度
2.复杂类型(如结构)的默认对齐方式是它最长的成员的对齐方式,这样在成员是复杂类型时,可以最小化长度
3.对齐后的长度必须是成员中最大的对齐参数的整数倍,这样在处理数组时可以保证每一项都边界对齐
补充一下,对于数组,比如:
char a[3];这种,它的对齐方式和分别写3个char是一样的.也就是说它还是按1个字节对齐.
如果写: typedef char Array3[3];
Array3这种类型的对齐方式还是按1个字节对齐,而不是按它的长度.
不论类型是什么,对齐的边界一定是1,2,4,8,16,32,64....中的一个.
整理后笔记:
规则:
1.每个成员按其类型的对齐参数(通常是这个类型的大小)和指定对齐参数(下面所示8字节)中较小的一个对齐
2.如果包含有结构体,那么按结构体中最大的倍数对齐
3.数组总大小要对齐到最大元素的倍数
#pragma pack(push, 8)
Struct s1
{
Short a;
long b;
};
Struct s2
{
char c;
S1 x;
long long lld;
};
#pragma pack(pop)
<1>sizeof(s2) = 24
<2>s2的c后面空了几个字节接着是x
(1)成员对齐有一个重要的条件,即每个成员分别按自己的方式(通常是这个类型的大小)对齐,也就是说上面虽然指定了按8字节对齐,但并不是所有的成员都是以8字节对齐
(2)对于结构来说,他的默认对齐方式就是它的所有成员使用的对齐参数中最大的一个,在s1中就是4
(3)对于最后一个成员lld是8个字节,和指定的一样,所以按8字节对齐,此时已经使用了12个字节,所以要添加4个字节,对齐到8字节倍数16字节上开始
(4)布局如下:
S1a b
11**,1111
S2c s1.a s1.b lld
1***,11**,1111,****11111111
对齐后的长度必须是成员中最大的对齐参数的整数倍,这样在处理数组时可以保证每一项都边界对齐,比如:char arr[3];这种对齐和分别写3个char一样按1字节对齐