关于结构体对齐

许多实际的计算机系统对基本类型数据在内存中存放的位置有限制,它们会要求这些数据的起始地址的值是某个数k的倍数,这就是所谓的内存对齐,而这个k则被称为该数据类型的对齐模数(alignment modulus)。这种强制的要求一来简化了处理器与内存之间传输系统的设计,二来可以提升读取数据的速度。 比如这么一种处理器,它每次读写内存的时候都从某个8倍数的地址开始,一次读出或写入8个字节的数据,假如软件能保证double类型的数据都从8倍数地址开始,那么读或写一个double类型数据就只需要一次内存操作。否则,我们就可能需要两次内存操作才能完成这个动作,因为数据或许恰好横跨在两个符合对齐要求的8字节内存块上。结构体对齐有以下两方面:

1)结构体总长度; 
2)结构体内各数据成员的内存对齐,即该数据成员相对结构体的起始位置;

在用sizeof运算符求算某结构体所占空间时,并不是简单地将结构体中所有元素各自占的空间相加,这里涉及到内存字节对齐的问题。从理论上讲,对于任何 变量的访问都可以从任何地址开始访问,但是事实上不是如此,实际上访问特定类型的变量只能在特定的地址访问,这就需要各个变量在空间上按一定的规则排列, 而不是简单地顺序排列,这就是内存对齐。

      内存对齐的原因:

      1)某些平台只能在特定的地址处访问特定类型的数据;

      2)提高存取数据的速度。比如有的平台每次都是从偶地址处读取数据,对于一个int型的变量,若从偶地址单元处存放,则只需一个读取周期即可读取该变量;但是若从奇地址单元处存放,则需要2个读取周期读取该变量。

       win32平台下的微软C编译器对齐策略:

      1)结构体变量的首地址能够被其最宽数据类型成员的大小整除。编译器在为结构体变量开辟空间时,首先找到结构体中最宽的数据类型,然后寻找内存地址能被该数据类型大小整除的位置,这个位置作为结构体变量的首地址。而将最宽数据类型的大小作为对齐标准。

      2)结构体每个成员相对结构体首地址的偏移量(offset)都是每个成员本身大小的整数倍,如有需要会在成员之间填充字节。编译器在为结构体成员开辟空 间时,首先检查预开辟空间的地址相对于结构体首地址的偏移量是否为该成员大小的整数倍,若是,则存放该成员;若不是,则填充若干字节,以达到整数倍的要 求。

      3)结构体变量所占空间的大小必定是最宽数据类型大小的整数倍。如有需要会在最后一个成员末尾填充若干字节使得所占空间大小是最宽数据类型大小的整数倍。

举例说明:

#pragma pack(8)

struct test1{

      int a;

      char b;

      int c[20]

long l;

} ;

struct test2{

      char a1;

      char a2;

      struct test1 t1;

      double b1;

}

#pragma pack(pop)

先计算test1, 8对齐,a占用0-3,b占用4,c占用8-87,l占用88-91,一共92个字节。成员中最大的对齐参数是int了92%4=0;

再计算test2, a1z占用0,a2占用1,t1呢,4 % 4 (test1里面最长的成员的对齐方式) = 0, 4-95,b1占96到103;一共104个字节,成员中最大的对齐参数是double了104%8=0; 所以是104.



 


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C语言中的结构体对齐是一个经常被问到的面试题。这里我来简单解释一下。 在C语言中,结构体是一种用户自定义的数据类型,它允许我们把多个不同类型的变量组合在一起,以便更方便地管理和使用它们。 在内存中,结构体的存储空间是按照成员变量的顺序依次分配的。但是,为了提高访问效率,编译器会对结构体进行对齐操作,也就是将结构体的起始地址调整为成员变量大小的整数倍。 例如,一个包含三个成员变量的结构体: ``` struct Test { char a; int b; short c; }; ``` 在32位系统中,char占1个字节,int占4个字节,short占2个字节,所以这个结构体的大小应该是1 + 4 + 2 = 7字节。但是,如果我们直接按顺序分配,结构体的起始地址是一个奇数,这样访问效率会降低。因此,编译器会在结构体的成员变量之间插入一些字节,使得结构体的起始地址是4的倍数。这样,结构体的大小就会变成12字节。 具体的对齐规则和字节数是由编译器决定的,不同的编译器可能会有不同的规则。但是,大多数编译器都会使用和本例相似的规则。 在面试中,经常会出现一些关于结构体对齐的问题,例如: 1. 结构体的大小是多少? 2. 结构体成员变量的顺序会影响结构体的大小吗? 3. 如何使用#pragma pack指令来控制结构体对齐? 4. 等等。 对于这些问题,我们需要对结构体对齐有一个深刻的理解,才能够正确回答。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值