结构体对齐

#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>s2c后面空了几个字节接着是x


(1)成员对齐有一个重要的条件,即每个成员分别按自己的方式(通常是这个类型的大小)对齐,也就是说上面虽然指定了按8字节对齐,但并不是所有的成员都是以8字节对齐

(2)对于结构来说,他的默认对齐方式就是它的所有成员使用的对齐参数中最大的一个,在s1中就是4

(3)对于最后一个成员lld8个字节,和指定的一样,所以按8字节对齐,此时已经使用了12个字节,所以要添加4个字节,对齐到8字节倍数16字节上开始

(4)布局如下:

S1a  b

  11**1111

S2c s1.a   s1.b  lld

  1***11**1111****11111111

对齐后的长度必须是成员中最大的对齐参数的整数倍,这样在处理数组时可以保证每一项都边界对齐,比如:char arr[3];这种对齐和分别写3char一样按1字节对齐


你可能感兴趣的:(c,结构体对齐)