C++中结构体的对齐方式

时间:2021-09-27 04:49:44

在面试中,常会考到结构体的对齐方式,因此对其进行总结。

1、在没有#pragma pack宏的情况下

struct sA{
	double d1;
	int i1;
	double d2;
	char c1;
};

 原则1:每个成员按类型的大小对齐,即相对于结构体地址的成员地址能被类型大小整除.并且结构体的大小(sizeof(A))必须为成员所含类型中最大值(sizeof(double))的整数倍,不够就补空字节.

例1:C++中结构体的对齐方式

例如用结构体定义对象A,其第一个成员d1的类型为double,大小为8字节,它的地址(也是A的地址)就需要能被4整除,这主要是因为为A申请内存时,其地址由其成员中所含的最大类型单独申请内存的规则决定(A中最大类型为double),int和double类型在单独申请内存时,其地址为4的整数倍。第二个成员i1的类型为int,大小为4字节,它相对于结构体A地址(00BDF7EC)的成员地址(8)能被4整除,不需要补充空白字节。第三个成员d2的类型为double,大小为8字节,它相对于结构体A地址(00BDF7EC)的成员地址(12)不能被8整除,需要补充4个空白字节。第四个成员c1的类型为char,大小为1字节,它相对于结构体A地址(00BDF7EC)的成员地址(16)能被1整除,不需要补充空白字节。

 原则2:结构体作为成员,类型大小按其成员所含最大类型计算。

struct sB{
	char c1;
	sA sA1;
};


例2:struct sB里存有struct sA,其结构体成员sA1的类型大小按8字节计算

 原则3:结构体的总大小,为其成员中所含最大类型的整数倍。在例1中即为成员d1类型(double,8)的整数倍。而在例二中即为成员sA1类型(8)的整数倍。

2、在有#pragma pack宏的情况下

(1)#pragma pack(1),例1和例2的情况如下

C++中结构体的对齐方式

可以看到其成员均按1作为其类型大小进行对齐,结构体的总大小也为1的整数倍。

(2)#pragma pack(2),例1和例2的情况如下

C++中结构体的对齐方式

可以看到其成员均按2作为其类型大小进行对齐,结构体的总大小也为2的整数倍。


(3)#pragma pack(4),例1和例2的情况如下

C++中结构体的对齐方式C++中结构体的对齐方式

可以看到其成员均按4作为其类型大小进行对齐,结构体的总大小也为4的整数倍。

附上代码

#include<stdio.h>
#include<iostream>
using namespace std;
#pragma pack(2)
struct sA{
	double d1;
	int i1;
	double d2;
	char c1;
};
struct sB{
	char c1;
	sA sA1;
};

int main()
{
	sA A;
	sB B;
	cout << "A的地址:" << &A << endl
		<< "A.d1的地址:" << &A.d1 << endl
		<< "A.i1的地址:" << &A.i1 << endl
		<< "A.d2的地址:" << &A.d2 << endl
		<< "A.c1的地址:" <<(void*)&A.c1 << endl
		<< "A的大小:" << sizeof(A) << endl;

	cout << "B的地址:" << &B << endl
		<< "B.c1的地址:" <<(void*) &B.c1 << endl
		<< "B.sA1的地址:" << &B.sA1 << endl
		<< "B的大小:" << sizeof(B) << endl;
	while (1);	
	return 0;
}