结构体对齐sizeof(struct x)

转载:http://blog.163.com/bbluesnow@126/blog/static/27784545201232125427164/

sizeof计算struct的大小  

2012-04-21 14:55:41|  分类: VC++ & MFC |  标签:结构体大小的定义  

 字节对齐的细节和编译器实现相关,但一般而言,满足三个准则:

1) 结构体变量的首地址能够被其最宽基本类型成员的大小所整除;

2) 结构体每个成员相对于结构体首地址的偏移量(offset)都是成员大小的整数倍,如有需要编译器会在成员之间加上填充字节(internal adding);

3) 结构体的总大小为结构体最宽基本类型成员大小的整数倍,如有需要编译器会在最末一个成员之后加上填充字节(trailing padding


类型

对齐方式(变量存放的起始地址相对于结构的起始地址的偏移量)

Char   偏移量必须为sizeof(char)1的倍数

int  偏移量必须为sizeof(int)4的倍数

float  偏移量必须为sizeof(float)4的倍数

double 偏移量必须为sizeof(double)8的倍数 

Short  偏移量必须为sizeof(short)2的倍数


简单说明:

struct MyStruct

{

char dda;        //  偏移量为0,满足对齐方式,dda占用1个字节;

double dda1;  //  下一个可用的地址的偏移量为1,不是sizeof(double)=8的倍数,需要补足7个字节才能使偏移量变为8(满     

                       //  足对 齐方式),因此VC自动填充7个字节,dda1存放在偏移量为8的地址上,它占用8个节。

int type;          // 下一个可用的地址的偏移量为16,是sizeof(int)=4的倍数,满足int的对齐方式,所以不需要VC自动填充,

                      // type/放在偏移量为16的地址上,它占用4个字节。

}         

所有成员变量都分配了空间,空间总的大小为1 7 8 4=20,不是结构的节边界数(即结构中占用最大空间的类型所占用的字节数sizeof 字串2(double)=8)的倍数,所以需要填充4个字节,以满足结构的大小为sizeof(double)=8的倍数。

所以该结构总的大小为:

sizeof(MyStruc)1 7 8 4 4=24。其中总的有7 4=11个字节是VC自动填充的,没有放任何有意义的东西。


VC 中提供了#pragma pack(n)来设定变量以n字节对齐方式。

1) n字节对齐就是说变量存放的起始地址的偏移量有两种情况:第一、如果n大于等于该变量所占用的字节数,那么偏 移量必须满足默认的对齐方式,第二、如果n小于该变量的类型所占用的字节数,那么偏移量为n的倍数,不用满足默认的对齐方式。

2) 结构的总大小也有个约束条 件,分下面两种情况:如果n大于所有成员变量类型所占用的字节数,那么结构的总大小必须为占用空间最大的变量占用的空间数的倍数;


下面举例说明其用法。

#pragma pack(push) //保存对齐状态

#pragma pack(4)//设定为4字节对齐

struct test

{

char m1;

double m4;

int m3;

};

#pragma pack(pop)//恢复对齐状态

   以上结构的大小为16,下面分析其存储情况,首先为m1分配空间,其偏移量为0,满足我们自己设定的对齐方式(4字节对齐),m1占用1个字节。接着开始为 m4分配空间,这时其偏移量为1,需要补足3个字节,这样使偏移量满足为n=4的倍数(因为sizeof(double)大于n,m4占用8个字节。接 着为m3分配空间,这时其偏移量为12,满足为4的倍数,m3占用4个字节。这时已经为所有成员变量分配了空间,共分配了16个字节,满足为n的倍数。如 果把上面的#pragma pack(4)改为#pragma pack(16),那么我们可以得到结构的大小为24

原文:忘记原文出处了, 汗!


例子:

//---测试sizeof(struct x)---

    // http://blog.163.com/bbluesnow@126/blog/static/27784545201232125427164/

    // http://blog.sina.com.cn/s/blog_5eb2e54e0100hh75.html

    struct A {

        char a;  //字长1对其系数1

        char b;  //字长1对其系数1

        char c;  //字长1对其系数1

    }; //整体对其系数为1

    NSLog(@"A = %ld", sizeof(struct A)); // 3 = 1 + 1 + 1

    

    struct B {

        int a;       //对其系数4

        char b;      //对其系数1

        short c;     //对其系数2

    }; //整体对其系数4

    NSLog(@"B = %ld", sizeof(struct B)); // 8 = 4 + 1 + (1 + 2)

    

    struct C {

        char b;   //对其系数1

        int a;      //对其系数4

        short c;  //对其系数2

    };//整体对其系数4

    NSLog(@"C = %ld", sizeof(struct C)); // 12 = (1 + 3) + 4 + 2 + 2(最大系数4n)

    

    struct ProtocolHead

    {

        uint16_t identitfy; // fixed with 0xabcd

        uint8_t  encrypt;

        uint32_t messageLen;

    };

    NSLog(@"ProtocolHead = %ld", sizeof(struct ProtocolHead)); // 8 = 2 + 1 + (1 + 4)

    

    struct MessageHead

    {

        uint8_t  phoneType; // 1 - iPhone, 2 - Android;

        uint16_t version;  // 0x0100 - 1.0, 0x0101 - 1.1 ;

        uint16_t messageType;

    };

    NSLog(@"MessageHead = %ld", sizeof(struct MessageHead)); // 6 = 1 + ( 1 + 2) + 2

    //---测试sizeof(struct x)---


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值