结构体大小的计算方法
先举三个例子
栗子1:
struct s1
{
char c1;
int i ;
char c2;
};
printf("%d\n",sizeof(struct s1));
栗子2:
struct s2
{
double d;
char c ;
int i;
};
printf("%d\n",sizeof(struct s2));
栗子3:
struct s3
{
char c1;
struct s2 s2 ;
int j;
};
printf("%d\n",sizeof(struct s3));
解决此题的关键是:结构体内存对齐
对齐规则:
1、第一个成员在与结构体变量偏移量为0的地址处;
2、其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
对齐数 = 编译器默认的一个对齐数与该成员大小的较小值
在vs中,默认对齐数是8
Linux环境下没有默认对齐数,对齐数为成员自身大小
3、结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍。
4、如果嵌套了结构体的情况(例3),嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍。
栗子1解答:
struct s1
{
char c1;
int i ;
char c2;
};
printf("%d\n",sizeof(struct s1));
由规则1和2可得
c1:min(1,8)= 1
i : min(4,8)= 4
c2 :min(1,8)=1
存储位置如图所示:
由图可知当前结构体已占用9字节大小,跟据规则3可得,结构体总大小为4*3=12(12 > 9)
栗子2解答:
struct s2
{
double d;
char c ;
int i;
};
printf("%d\n",sizeof(struct s2));
由规则1和2可得
d:min(8,8)= 8
c :min(1,8)=1
i : min(4,8)= 4
存储位置如图所示:
由图可知当前结构体已占用16字节大小,跟据规则3可得,结构体总大小为8*2=16
栗子3解答:
struct s3
{
char c1;
struct s2 s2 ;
int j;
};
printf("%d\n",sizeof(struct s3));
由规则1,2和 3 可得
c1 :min(1,8)=1
struct s2中
d:min(8,8)= 8
c :min(1,8)=1
i : min(4,8)= 4
得s2最大对齐数为8,所以s2数据从位置8开始存储
j: min(4,8)= 4
存储位置如图所示:
由图可知当前结构体已占用28字节大小,跟据规则3可得,结构体总大小为8*4=32(32 > 28)