结构体对齐(64位Linux)

结构体对齐(64位Linux)

有关结构体对齐最关键的是要理清三个数值:

  • 自身对齐值:变量的类型原有的默认对齐值,比如int是4,double是8,char是1
  • 指定对齐值:通过#pragma pack或者attribute语法来显式地i指定对齐值
  • 有效对齐值:Min(指定对齐值,自身对齐值)

结构体或类的自身对齐值为其最大成员自身对齐值

下面上代码分析:

#include <unistd.h>
#include <iostream>

using namespace std;

class unpacked
{
public:
    int a;
    char b;
    double c;
};

class PackedByAttrribute
{
public:
    int a;
    char b;
    double c;
} __attribute__((packed));

#pragma pack(2)
class PackedByPragma
{
public:
    int a;
    char b;
    double c;
};
#pragma pack()

class Mix
{
public:
    unpacked a;
    PackedByAttrribute b;
    PackedByPragma c;
    int d;
    char e;
    double f;
    char g;
};
//or:

int main(int argc, char **argv)
{
    unpacked A;
    PackedByAttrribute B;
    PackedByPragma C;
    Mix D;
    printf("unpacked: %d a:%p,b:%p,c:%p
", sizeof(A), &A.a, &A.b, &A.c);
    printf("packed by attribute: %d a:%p,b:%p,c:%p
", sizeof(B), &B.a, &B.b, &B.c);
    printf("packed by pragma pack: %d a:%p,b:%p,c:%p
", sizeof(C), &C.a, &C.b, &C.c);
    printf("Mix: %d a:%p,b:%p,c:%p,d:%p,e:%p,f:%p,g:%p
", sizeof(D), &D.a, &D.b, &D.c, &D.d, &D.e, &D.f, &D.g);
    return 0;
}

输出:

unpacked: 16 a:0x7fff41d5c9b0,b:0x7fff41d5c9b4,c:0x7fff41d5c9b8
packed by attribute: 13 a:0x7fff41d5c995,b:0x7fff41d5c999,c:0x7fff41d5c99a
packed by pragma pack: 14 a:0x7fff41d5c9a2,b:0x7fff41d5c9a6,c:0x7fff41d5c9a8
Mix: 72 a:0x7fff41d5c9c0,b:0x7fff41d5c9d0,c:0x7fff41d5c9de,d:0x7fff41d5c9ec,e:0x7fff41d5c9f0,f:0x7fff41d5c9f8,g:0x7fff41d5ca00

分析

unpacked

gcc在64位下默认8字节对齐,最后一个double需要8地址对齐,所以需要空多3字节。这个类的有效对齐值为8,因为其成员的最大对齐值是8。

packedByAttribute

这是gcc的编译器特有的语法,这样指定相当于1字节对齐,也就是pack(1),也有aligned(n)的语法。网络编程中时不时遇到,这样做可以节省一点网络流量,忽略双方机器的系统对齐情况。其大小就是各成员大小相加。

packedByPack(2)

指定2字节对齐,所以最后一个double变量仅需要空一个字节便可实现2字节对齐,其有效对齐值为2。

Mix

仅需要注意最后的char变量因为整个类的有效对齐值为8,所以整个类的大小为其对齐值的整数倍,所以还要补充7个字节对齐,各个成员在Mix中都根据自己的有效对齐值来对齐。

原文地址:https://www.cnblogs.com/manch1n/p/14960290.html