Linux下C语言——结构体对齐

结构体对齐的步骤:

            1、结构体各成员对齐

            2、整个结构体圆整

结构体对齐的特定对齐值:

            1、自身对齐值: 自身对齐值就是结构体变量里每个成员的自身大小;

            2、指定对齐值: 指定对齐值是由宏#pragma pack (N)指定的值,里面的N必须是2的幂次方,如1,2,4,8,16等。如果没有通过#pragma pack宏那么在32位的Linux主机上默认指定对齐值为4,64位的Linux主机上默认指定对齐值为8,ARM CPU默认指定对齐值为8;

            3、有效对齐值: ① 结构体成员自身对齐时有效对齐值为自身对齐值与指定对齐值中较小的一个,
                                       ② 结构体圆整时,为所有成员中自身对齐值最大的与指定对齐值较小的一个;

下面举两个简单的例子(32位系统):

typedef struct _st_struct1			typedef struct _st_struct2
{									{
    char       	  a;					short      	  b;
    short         b;					int           c;
    int           c;					char          a;
}st_struct1;						}st_struct2;

        这两个结构体内部的数据类型是相同的,那是不是也意味着这两个结构体所占用的存储空间也是相同的呢?答案是不同的。

        我们先运用上面讲解的流程来对这两个结构体分别进行对齐。首先将确定自身对齐值以及指定对齐值:

  • 自身对齐值:char  1byte、short   2bytes、int  4bytes
  • 指定对齐值:4bytes(指定为32位系统)

下面对结构体st_struct1进行对齐:

在这里插入图片描述
下面对结构体st_struct2进行对齐:
在这里插入图片描述
        可见虽然含有的数据类型相同,但是结构体st_struct1和结构体st_struct2所占用的存储空间却不同。

        以上就是结构体对齐的简单举例。

        另外需要注意的是,Windows操作系统与Linux操作系统关于结构体对齐会有一点差别,Windows操作系统中在结构体对齐的第二步也就是整个结构体圆整时,所用的是成员自身对齐值中最大的那一个。这个小的差别是在牛客网上做笔试题时发现的。

//32位机器上定义如下结构体:
struct xx
{
    long long _x1;
    char _x2;
    int _x3;
    char _x4[2];
    static int _x5;	
    //注:_x5为静态变量,程序未编译时就存在,其存储在代码数据段中的bss区中
};
int xx::_x5;

//请问sizeof(xx)的大小是()
A、19	B、20	C、15	D、24

按照前面所讲的,并且在Linux下编译通过后,得到的结果是20(4字节对齐)。同样也在Windows下编译通过后,得到的结果是24(8字节对齐)。

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、付费专栏及课程。

余额充值