内存对齐(#pragma pack(n))

本文讨论限制在VC++ 编译器


讨论对象:

#pragma pack([n])


1. 对齐原则为(参考百度百科#pragma pack词条):

编译器中提供了#pragma pack(n)来设定变量以n字节对齐方式。n字节对齐就是说变量存放的起始地址的偏移量有两种情况:第一、如果n大于等于该变量所占用的字节数,那么偏移量必须满足默认的对齐方式,第二、如果n小于该变量的类型所占用的字节数,那么偏移量为n的倍数,不用满足默认的对齐方式。结构的总大小也有个约束条件,分下面两种情况:如果n大于所有成员变量类型所占用的字节数,那么结构的总大小必须为占用空间最大的变量占用的空间数的倍数;否则必须为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分配空间,这时其 偏移量为4,需要补足3个字节,这样使偏移量满足为n=4的倍数(因为sizeof(double)大于4),m4占用8个字节。接着为m3分配空间,这时其 偏移量为12,满足为4的倍数,m3占用4个字节。这时已经为所有 成员变量分配了空间,共分配了16个字节,满足为n的倍数。如果把上面的 #pragma pack(4)改为 #pragma pack(8),那么我们可以得到结构的大小为24。
补充一点: 如果写成"#pragma pack()" 即,n 值 没写, 则表示按默认对齐

2. 内嵌结构体

#include <stdio.h>

typedef unsigned char UINT8;
typedef unsigned int UINT32;

#pragma pack(4)
 
typedef struct _xxx {
    UINT8 a;
    UINT32 b;
}xxx;
 
#pragma pack()
 
 
 
 
#pragma pack(2)
typedef struct _yyy {
    UINT8 a;
    xxx b;            //这里的内存对齐怎么算???
} yyy;
#pragma pack()

void main()
{
	yyy y;
	yyy *py = &y;
	xxx *px;
	y.a = 1;
	y.b.a = 2;
	y.b.b = 3;

	px = &y.b;
	printf("%d-%d\n", px->a, px->b);
}

运行结果为

即:

结构体yyy 里xxx(b)的内部还是按xxx定义时的规则进行内存对齐. 然后在yyy里xxx(b)只是一个变量, 对齐要求为xxx里成员在对齐要求最大的数值做为xxx的对齐要求

再来一例, 注意看图片里红色方框里的cc(即未初始化的内容)

#include <stdio.h>


typedef unsigned char UINT8;
typedef unsigned int UINT32;


#pragma pack(4)
 
typedef struct _xxx {
    UINT8 a;
    UINT32 b;
    UINT8 c;
}xxx;
 
#pragma pack()
 
 
 
 
#pragma pack(2)
typedef struct _yyy {
    UINT8 a;
    xxx b;            //这里的内存对齐怎么算???
    UINT8 c;
} yyy;
#pragma pack()


#pragma pack(1)


typedef struct _zzz {
    UINT8 a;
    yyy b;
    UINT8 c;
} zzz;


#pragma pack()


void main()
{
    zzz z[2];
    yyy *py = &z[0].b;
    z->a = 1;
    z->b.a = 2;
    z->b.b.a = 3;
    z->b.b.b = 4;
    z->b.b.c = 5;
    z->b.c = 6;
    z->c = 7;
    memcpy(z+1, z, sizeof(zzz));
    printf("%d-%d\n", py->a, py->b.a);
}


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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值