64,646
社区成员
发帖
与我相关
我的任务
分享
class A
{
public:
double d;
int a;
float b;
char c;
A();
~A();
}
class A
{
public:
double d;
int a;
char c;
float b;
A();
~A();
};
这里没有指定对齐值(就是没有#pragma pack (n))
那么这个类的的最终的对齐值是 8,就是按照8字节对齐
假定 地址空间从0x0000开始排放,
那么double d的地址为 0x0000 (0x0000 % 8 = 0)
然后往后偏8字节之后,能被下一个成员int a这4字节对齐值整除的地址为0x0008
那么int a的地址为 0x0008 (0x0008 % 4 = 0)
然后往后偏4字节之后,能被char c这1字节对齐值整除的地址为0x000C
那么char c的地址为 0x000C (0x000C % 1 = 0)
然后往后偏1字节之后,能被float b这4字节对齐值整除的地址为0x0010(此时char c后面空出来了3字节)
那么float b的地址为 0x0010 (0x0010 % 4 = 0)[/quote]
太谢谢您了,我终于明白了。#include <stdio.h>
#define field_offset(s,f) (int)(&(((struct s *)(0))->f))
struct AD { int a; char b[13]; double c;};
#pragma pack(push)
#pragma pack(1)
struct A1 { int a; char b[13]; double c;};
#pragma pack(2)
struct A2 { int a; char b[13]; double c;};
#pragma pack(4)
struct A4 { int a; char b[13]; double c;};
#pragma pack(8)
struct A8 { int a; char b[13]; double c;};
#pragma pack(16)
struct A16 { int a; char b[13]; double c;};
#pragma pack(pop)
int main() {
printf("AD.a %d\n",field_offset(AD,a));
printf("AD.b %d\n",field_offset(AD,b));
printf("AD.c %d\n",field_offset(AD,c));
printf("\n");
printf("A1.a %d\n",field_offset(A1,a));
printf("A1.b %d\n",field_offset(A1,b));
printf("A1.c %d\n",field_offset(A1,c));
printf("\n");
printf("A2.a %d\n",field_offset(A2,a));
printf("A2.b %d\n",field_offset(A2,b));
printf("A2.c %d\n",field_offset(A2,c));
printf("\n");
printf("A4.a %d\n",field_offset(A4,a));
printf("A4.b %d\n",field_offset(A4,b));
printf("A4.c %d\n",field_offset(A4,c));
printf("\n");
printf("A8.a %d\n",field_offset(A8,a));
printf("A8.b %d\n",field_offset(A8,b));
printf("A8.c %d\n",field_offset(A8,c));
printf("\n");
printf("A16.a %d\n",field_offset(A16,a));
printf("A16.b %d\n",field_offset(A16,b));
printf("A16.c %d\n",field_offset(A16,c));
printf("\n");
return 0;
}
//AD.a 0
//AD.b 4
//AD.c 24
//
//A1.a 0
//A1.b 4
//A1.c 17
//
//A2.a 0
//A2.b 4
//A2.c 18
//
//A4.a 0
//A4.b 4
//A4.c 20
//
//A8.a 0
//A8.b 4
//A8.c 24
//
//A16.a 0
//A16.b 4
//A16.c 24
//
//
class A
{
public:
double d;
int a;
char c;
float b;
A();
~A();
};
这里没有指定对齐值(就是没有#pragma pack (n))
那么这个类的的最终的对齐值是 8,就是按照8字节对齐
假定 地址空间从0x0000开始排放,
那么double d的地址为 0x0000 (0x0000 % 8 = 0)
然后往后偏8字节之后,能被下一个成员int a这4字节对齐值整除的地址为0x0008
那么int a的地址为 0x0008 (0x0008 % 4 = 0)
然后往后偏4字节之后,能被char c这1字节对齐值整除的地址为0x000C
那么char c的地址为 0x000C (0x000C % 1 = 0)
然后往后偏1字节之后,能被float b这4字节对齐值整除的地址为0x0010(此时char c后面空出来了3字节)
那么float b的地址为 0x0010 (0x0010 % 4 = 0)class A
{
public:
double d;
int a;
char c;
float b;
A();
~A();
}
class A
{
};
少个分号
1) 结构体变量的首地址能够被其最宽基本类型成员的大小所整除;
2) 结构体每个成员相对于结构体首地址的偏移量(offset)都是成员大小的整数倍,如有需要编译器会在成员之间加上填充字节(internal adding);
3) 结构体的总大小为结构体最宽基本类型成员大小的整数倍,如有需要,编译器会在最末一个成员之后加上填充字节(trailing padding)。