引用:内存字节对齐

写出一个struct,然后sizeof,你会不会经常对结果感到奇怪?sizeof的结果往往都比你声明的变量总长度要大,这是怎么回事呢?讲讲字节对齐吧.

/******************************分割线

如果体系结构是不对齐的,A中的成员将会一个挨一个存储,从而sizeof(a)为11。显然对齐更浪费了空间。那么为什么要使用对齐呢?

体系结构的对齐和不对齐,是在时间和空间上的一个权衡。对齐节省了时间。假设一个体系结构的字长为w,那么它同时就假设了在这种体系结构上对宽度为w的数据的处理最频繁也是最重要的。它的设计也是从优先提高对w位数据操作的效率来考虑的。比如说读写时.............此处省略50万字

***********************************************************/

上面是你随便 google一下,人家就可以跟你解释的,一大堆的道理,我们没怎么多时间,讨论为何要对齐.直入主题,怎么判断内存对齐规则,sizeof的结果怎么来的,请牢记以下3条原则:(在没有#pragma pack宏的情况下,务必看完最后一行)

1:数据成员对齐规则:结构(struct)(或联合(union))的数据成员,第一个数据成员放在offset为0的地方,以后每个数据成员存储的起始位置要从该成员大小或者成员的子成员大小(只要该成员有子成员,比如说是数组,结构体等)的整数倍开始(比如int在32位机为4字节,则要从4的整数倍地址开始存储。

2:结构体作为成员:如果一个结构里有某些结构体成员,则结构体成员要从其内部最大元素大小的整数倍地址开始存储.(struct a里存有struct b,b里有char,int ,double等元素,那b应该从8的整数倍开始存储.)

3:收尾工作:结构体的总大小,也就是sizeof的结果,.必须是其内部最大成员的整数倍.不足的要补齐.

4:注意套用三原则里计算的对齐值是不能大于#pragma pack指定的n值. gcc默认是#pragma pack(4),并且gcc只支持1,2,4对齐。

等你看完此3条原则,2分钟已经过去,抓紧时间,实战3分钟:

typedef struct bb

{

int id;             //[0]....[3]

double weight;      //[8].....[15]      原则1

float height;      //[16]..[19],总长要为8的整数倍,补齐[20]...[23]     原则3//注意4字节对其的编译器里面最大基数为4。

}BB;

typedef struct aa

{

char name[2];     //[0],[1]

int  id;         //[4]...[7]          原则1

double score;     //[8]....[15]

short grade;    //[16],[17]

BB b;             //[24]......[47]          原则2

}AA;

int main()

{

AA a;

cout<<sizeof(a)<<" "<<sizeof(BB)<<endl;

return 0;

}

结果是

48 24

ok,上面的全看明白了,内存对齐基本过关.

再讲讲#pragma pack().

在代码前加一句#pragma pack(1),你会很高兴的发现,上面的代码输出为

32 16

bb是4+8+4=16,aa是2+4+8+2+16=32;

这不是理想中的没有内存对齐的世界吗.没错,#pragma pack(1),告诉编译器,所有的对齐都按照1的整数倍对齐,换句话说就是没有对齐规则.

明白了不?

那#pragma pack(2)的结果又是多少呢?对不起,5分钟到了,自己去测试吧.

ps:Vc,Vs等编译器默认是#pragma pack(8),所以测试我们的规则会正常;注意gcc默认是#pragma pack(4),并且gcc只支持1,2,4对齐。套用三原则里计算的对齐值是不能大于#pragma pack指定的n值。

源文档 <http://blog.csdn.net/hairetz/article/details/4084088>

测试:

// demo.cpp : 定义控制台应用程序的入口点。
// vs2010 #include "stdafx.h"
#include "stdlib.h"
#include <stdio.h> typedef struct bb
{
int id; //[0]....[3]
double weight; //[8].....[15]      原则1
float height; //[16]..[19],总长要为8的整数倍,补齐[20]...[23]     原则3//注意4字节对其的编译器里面最大基数为4。
}BB; typedef struct aa
{
char name[2]; //[0],[1]
int id; //[4]...[7]          原则1
double score; //[8]....[15]
short grade;//[16],[17]
BB b; //[24]......[47]          原则2
}AA; typedef struct zz
{
char name[2]; //[0],[1]
int id; //[4]...[7]          原则1
double score; //[8]....[15]
short grade;//[16],[17]
BB b; //[24]......[47]          原则2
char a;
}ZZ; int _tmain(int argc, _TCHAR* argv[])
{
AA a;
BB b;
ZZ z; printf("sizeof(a) = %d, sizeof(BB) = %d\n",sizeof(a),sizeof(BB));
printf("sizeof(z) = %d\n",sizeof(z)); printf("size max = double = %d\n",sizeof(double) ); system("pause"); return 0;
} /**
sizeof(a) = 48, sizeof(BB) = 24
sizeof(z) = 56
size max = double = 8
请按任意键继续. . .
**/

  

// demo.cpp : 定义控制台应用程序的入口点。
// linux #include <stdio.h> typedef struct bb
{
int id; //[0]....[3]
double weight; //[8].....[15]      原则1
float height; //[16]..[19],总长要为8的整数倍,补齐[20]...[23]     原则3//注意4字节对其的编译器里面最大基数为4。
}BB; typedef struct aa
{
char name[2]; //[0],[1]
int id; //[4]...[7]          原则1
double score; //[8]....[15]
short grade;//[16],[17]
BB b; //[24]......[47]          原则2
}AA; typedef struct zz
{
char name[2]; //[0],[1]
int id; //[4]...[7]          原则1
double score; //[8]....[15]
short grade;//[16],[17]
BB b; //[24]......[47]          原则2
char a;
}ZZ; int main(int argc, char* argv[])
{
AA a;
BB b;
ZZ z; printf("sizeof(a) = %d, sizeof(BB) = %d\n",sizeof(a),sizeof(BB));
printf("sizeof(z) = %d\n",sizeof(z)); printf("size max = double = %d\n",sizeof(double) ); return 0;
} /**
root@oucaijun:/work/dcc# gcc *.c; ./a.out
sizeof(a) = 36, sizeof(BB) = 16
sizeof(z) = 40
size max = double = 8
**/

  

c语言,内存字节对齐的更多相关文章

  1. C语言:内存字节对齐详解[转载]

    一.什么是对齐,以及为什么要对齐: 1. 现代计算机中内存空间都是按照byte划分的,从理论上讲似乎对任何类型的变量的访问可以从任何地址开始,但实际情况是在访问特定变量的时候经常在特定的内存地址访问, ...

  2. C语言:内存字节对齐详解

    转:http://blog.csdn.net/arethe/article/details/2548867 一.什么是对齐,以及为什么要对齐: 1. 现代计算机中内存空间都是按照byte划分的,从理论 ...

  3. 【C语言】字节对齐(内存对齐)

    数据对齐 1.  对齐原则: [原则1]数据成员对齐规则:结构(struct)(或联合(union))的数据成员,第一个数据成员放在offset为0的地方,以后每个数据成员的对齐按照#pragma p ...

  4. 转载:C语言的字节对齐及#pragma pack的使用

    C语言的字节对齐及#pragma pack的使用   C编译器的缺省字节对齐方式(自然对界) 在缺省情况下,C编译器为每一个变量或是数据单元按其自然对界条件分配空间. 在结构中,编译器为结构的每个成员 ...

  5. C++:struct和union 内存字节对齐问题

    转自:http://blog.csdn.net/wangyanguiyiyang/article/details/53312049 struct内存对齐问题 1:数据成员对齐规则:结构(struct) ...

  6. C++内存字节对齐规则

    为什么要进行内存对齐以及对齐规则 C/C++—— 内存字节对齐规则 C++内存字节对齐规则

  7. C语言之字节对齐

    在C语言编程中,有时为了达到减少运行的时间的目的,需要浪费一些空间:而有时为了节省空间,使它的运行时间增长.而字节对齐则是为了访问效率,用空间换取时间. 要掌握字节对齐,首先得明确一下四个概念: 1. ...

  8. struct/class等内存字节对齐问题详解

    问题引入 定义一个结构体的一般形式为: struct 结构体名 { //类型说明符 成员名; }; 例如有如下结构体: struct Stu { int id; char sex; float hig ...

  9. 【C语言】字节对齐问题(以32位系统为例)

    1. 什么是对齐? 现代计算机中内存空间都是按照字节(byte)划分的,从理论上讲似乎对任何类型的变量的访问可以从任何地址开始,但实际情况是在访问特定变量的时候经常在特定的内存地址访问,这就需要各类型 ...

随机推荐

  1. AnyEvent::HTTP 介绍

    AnyEvent::HTTP - simple but non-blocking HTTP/HTTPS client 一个简单的非堵塞的 HTTP/HTTPS 客户端: use AnyEvent::H ...

  2. BZOJ 1611: [Usaco2008 Feb]Meteor Shower流星雨

    1611: [Usaco2008 Feb]Meteor Shower流星雨 Description 去年偶们湖南遭受N年不遇到冰冻灾害,现在芙蓉哥哥则听说另一个骇人听闻的消息: 一场流星雨即将袭击整个 ...

  3. css中border-width 属性

    border-width属性可能的值 值 描述 thin 定义细的边框. medium 默认.定义中等的边框. thick 定义粗的边框. length 允许您自定义边框的宽度. inherit 规定 ...

  4. UML 顺序图

    顺序图 顺序图是交互图的一种形式,它显示对象沿生命线发展,对象之间随时间的交互表示为从源生命线指向目标生命线的消息.顺序图能很好地显示那些对象与其它那些对象通信,什么消息触发了这些通信,顺序图不能很好 ...

  5. C#_会员管理系统:开发二(会员资料管理界面的‘增删改查’)

    会员资料管理界面: 新建一个窗体,窗体界面和控件如下: 窗体中的控件dgvManager更改FullRowSelect属性(点击选中效果)为:FullRowSelect 会员资料管理界面窗体的详细代码 ...

  6. BZOJ 3236: [Ahoi2013]作业( 莫队 + BIT )

    莫队..用两个树状数组计算.时间复杂度应该是O(N1.5logN). 估计我是写残了...跑得很慢... ----------------------------------------------- ...

  7. Android百度地图之显示地图

    添加地图显示 一.在百度官网下载相关的SDK (网址:http://developer.baidu.com/map/sdkandev-download.htm) 解压下载好的BaiduMap_Andr ...

  8. ibatis3.0调用Oracle的存储过程

    直接上源码 一,oracle储存过程. create or replace procedure proc_get_th(i_hth in varchar2,o_ret out sys_refcurso ...

  9. USACO Palindromic Squares 【STL__string_的应用】

    这里有个讲解 string 用法非常详细的博文:https://www.byvoid.com/zhs/blog/cpp-string 题目意思很简单啦,就是找回文 使用string可以高速A过 Sou ...

  10. 关于QuartusII对ram块的综合

    之前在看Altera的官方教程上就有说明,如果我们定义一个reg [`word_w]user_ram[`word_d]  ; QuartusII会自动综合成为一个ram—— 当然有一些前提:(后续补充 ...