C/C++中的结构体的字节对齐和#paragma pack (n)编译命令
2008-08-31 14:17
531 查看
结构体的字节对齐方式
在C语言中,结构是一种复合数据类型,其构成元素既可以是基本数据类型(如int、long、float等)的变量,也可以是一些复合数据类型(如数组、结构、联合等)的数据单元。在结构中,编译器为结构的每个成员按其自然对齐边界(alignment)条件分配空间。各个成员按照它们被声明的顺序在内存中顺序存储,第一个成员的地址和整个结构的地址相同。如下结构:
typedef struct _TEST{
char c1;
short s;
float f;
char c2;
}TEST;
结构的第一个成员c1,其偏移地址为0,占据了第1个字节。第二个成员s为short类型,其起始地址必须2字节对齐,因此,编译器在c1和s之间填充了一个空字节。结构的第三个成员f和第四个成员c2恰好落在其自然对齐边界地址上,在它们前面不需要额外的填充字节。在TEST结构中,成员f要求4字节对界,是该结构所有成员中要求的最大对齐边界单元,因而整个TEST结构的自然对齐边界为4字节,编译器在成员c2后面填充了3个空字节。整个结构所占据空间为12字节。
测试代码如下:
#include <stdio.h>
typedef struct _TEST{
char c1;
short s;
float f;
char c2;
}TEST,*PTEST;
int main(){
int i;
char* p;
TEST t = {'a',1,2.0,'b'};
printf("The base adress of the struct t: &t = 0x%xd/n", &t);
printf("The first element address: &t.c1 = 0x%xd/n", &t.c1);
printf("The size of the struct t: sizeof(t) = %d/n", sizeof(t));
printf("The memory of the struct: ");
p = (char*)&t;
for( i = 0; i < sizeof(t); printf("%x ",p[i++]));
printf("/n");
return 0;
}
Output:
The base adress of the struct t: &t = 0x12ff6cd
The first element address: &t.c1 = 0x12ff6cd
The size of the struct t: sizeof(t) = 12
The memory of the struct: 61 ffffffcc 1 0 0 0 0 40 62 ffffffcc ffffffcc ffffffcc
从测试代码的运行结果中我们可以看到,结构体的基址和第一个成员的地址是一样的。根据结构体的字节对齐规则,结构体的总字节数 sizeof(t) = 12,通过打印结构体t的内存情况我们可以看到它在第一个成员c1后填充了一个字节,在第三个成员后填充了3个字节。
2. 更改C编译器的缺省字节对齐方式
在缺省情况下,C编译器为每一个变量或是数据单元按其自然对界条件分配空间。一般地,可以通过下面的方法来改变缺省的对界条件:
使用伪指令#pragma pack (n),C编译器将按照n个字节对齐。
使用伪指令#pragma pack (),取消自定义字节对齐方式。
另外,还有如下的一种方式:
__attribute((aligned (n))),让所作用的结构成员对齐在n字节自然边界上。如果结构中有成员的长度大于n,则按照最大成员的长度来对齐。
__attribute((packed)),取消结构在编译过程中的优化对齐,按照实际占用字节数进行对齐。
同样是上面这个TEST结构,我们用#pragma pack(1)指令后,即
#pragma pack(1)
typedef struct _TEST{
char c1;
short s;
float f;
char c2;
}TEST,*PTEST;
#pragma()
//sizeof(TEST) = 8
在C语言中,结构是一种复合数据类型,其构成元素既可以是基本数据类型(如int、long、float等)的变量,也可以是一些复合数据类型(如数组、结构、联合等)的数据单元。在结构中,编译器为结构的每个成员按其自然对齐边界(alignment)条件分配空间。各个成员按照它们被声明的顺序在内存中顺序存储,第一个成员的地址和整个结构的地址相同。如下结构:
typedef struct _TEST{
char c1;
short s;
float f;
char c2;
}TEST;
结构的第一个成员c1,其偏移地址为0,占据了第1个字节。第二个成员s为short类型,其起始地址必须2字节对齐,因此,编译器在c1和s之间填充了一个空字节。结构的第三个成员f和第四个成员c2恰好落在其自然对齐边界地址上,在它们前面不需要额外的填充字节。在TEST结构中,成员f要求4字节对界,是该结构所有成员中要求的最大对齐边界单元,因而整个TEST结构的自然对齐边界为4字节,编译器在成员c2后面填充了3个空字节。整个结构所占据空间为12字节。
测试代码如下:
#include <stdio.h>
typedef struct _TEST{
char c1;
short s;
float f;
char c2;
}TEST,*PTEST;
int main(){
int i;
char* p;
TEST t = {'a',1,2.0,'b'};
printf("The base adress of the struct t: &t = 0x%xd/n", &t);
printf("The first element address: &t.c1 = 0x%xd/n", &t.c1);
printf("The size of the struct t: sizeof(t) = %d/n", sizeof(t));
printf("The memory of the struct: ");
p = (char*)&t;
for( i = 0; i < sizeof(t); printf("%x ",p[i++]));
printf("/n");
return 0;
}
Output:
The base adress of the struct t: &t = 0x12ff6cd
The first element address: &t.c1 = 0x12ff6cd
The size of the struct t: sizeof(t) = 12
The memory of the struct: 61 ffffffcc 1 0 0 0 0 40 62 ffffffcc ffffffcc ffffffcc
从测试代码的运行结果中我们可以看到,结构体的基址和第一个成员的地址是一样的。根据结构体的字节对齐规则,结构体的总字节数 sizeof(t) = 12,通过打印结构体t的内存情况我们可以看到它在第一个成员c1后填充了一个字节,在第三个成员后填充了3个字节。
2. 更改C编译器的缺省字节对齐方式
在缺省情况下,C编译器为每一个变量或是数据单元按其自然对界条件分配空间。一般地,可以通过下面的方法来改变缺省的对界条件:
使用伪指令#pragma pack (n),C编译器将按照n个字节对齐。
使用伪指令#pragma pack (),取消自定义字节对齐方式。
另外,还有如下的一种方式:
__attribute((aligned (n))),让所作用的结构成员对齐在n字节自然边界上。如果结构中有成员的长度大于n,则按照最大成员的长度来对齐。
__attribute((packed)),取消结构在编译过程中的优化对齐,按照实际占用字节数进行对齐。
同样是上面这个TEST结构,我们用#pragma pack(1)指令后,即
#pragma pack(1)
typedef struct _TEST{
char c1;
short s;
float f;
char c2;
}TEST,*PTEST;
#pragma()
//sizeof(TEST) = 8
相关文章推荐
- 结构体字节对齐,pragma pack,__attribute__(packed)
- C/C++ 结构体字节对齐详解
- C++ 字节对齐与结构体/共用体的大小
- [C++基础]内存对齐(字节对齐,结构体对齐)
- C/C++中结构体的字节对齐以及求sizeof()
- C/C++结构体字节对齐详解
- 关于C++ 字节对齐 (结构体)
- 结构体字节对齐,pragma pack,__attribute__(packed)
- C++ 结构体字节对齐问题
- C++中结构体的字节对齐问题 ---- 转
- C++中结构体的字节对齐问题
- #pragma pack 字节对齐预编译指令说明
- C++ 结构体字节对齐
- mapview中引入 TILEINFO结构体,忘记写#pragma pack(1)字节对齐,差点搞死。。。
- struct 结构体 、 union 枚举 及 #pragma pack 字节对齐
- 【c++】【转】结构体字节对齐
- 深入剖析C++中的struct结构体字节对齐
- C/C++ 结构体 字节对齐原则详细举例解释 及sizeof的基本用法
- C/C++基础之:结构体的字节对齐问题
- (c/c++学习笔记三) 结构体与联合体的字节对齐