C语言内存对齐问题小结

内存对齐:结构体成员存放的地址要能整除该成员本身的大小
内存对齐的原则:
1.前面所有成员的大小相加,应该是当前成员大小的倍数
2.整个结构体的大小应该能整除该结构体中单个最大成员的倍数

*ps 每个数据类型的大小注释在后面

举个栗子①

struct A
{
 char a;//1+1
 short b;//2
 int c;//4
};//8

定义一个结构体A,根据第一条原则,需要将成为short类型的大小的倍数,则需要原有大小1的情况下再加1。接着向下进行,char加上short大小1+1+2正好等于int的大小4,则此结构体大小为(1+1+2+4)=8

那么再来一个栗子②

struct C
{
 int a;//4
 char b;//1 +3
 short c;//2
};//8

这次定义一个结构体C,第一行4是1倍数,接着向下 4+1不是2倍数 所以这时应该是4+1+3而不是4+1+2,虽然同为2的倍数,但还需满足可整除该结构体中单个最大成员的倍数,而4是该结构体中单个成员最大的,要满足是他的倍数只能是4+1+3=8

那么那么,在栗子②的情况下,迎来了我们的栗子③

struct H
{
 char a;//1+3
 struct HH
 {
  int b;//4
  long c;//4
 }d;//8
};

这里定义一个结构体H,且在此结构体中再定义一个结构体HH,结构体HH大小为8,而此时为了满足原则1,需要对char进行+操作,参考单个最大成员int,则只需+3,此时结构体H大小为1+3+8=12

栗子④

struct I
{
 char a;//1
 struct II
 {
  int b;
  long c;
 };
};//1 

是1的原因为没有定义变量 只有char

栗子⑤

struct J
{
 char a;//1+3
 struct 
 {
  int b;//4
  long c;//4
 };//8
};//12

此处在结构体J中定义了一个没有命名的变量

#pragma pack()
除此还可以使用#pragma pack()来改变编译器的默认对齐规则,使用指#pragma pack(n),编译器将按照n个字节对齐,该指令还有一个条件,虽然指定了按照n个字节对齐,但是并不是每个成员都按照那个字节对齐,每个成员是按自己的方式进行对齐。

再来一个栗子

#pragma pack(8)
struct Z
{
       char c1;//1
        int i;//4
};

Z中char型变量为1个字节,按照对齐规则,选取min(1,8)=1,所以c按1字节对齐,int型为4个字节,min(8,4)=4,按4字节对齐,所以char型,即变量c后应该补三个字节.最终整个大小为8个字节,刚好是8的倍数(对齐后的长度是成员中最大的对齐参数的整数倍),不用进行字节补齐.

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值