探索结构体的内存对齐

 先上两个简单的结构体,思考:这两个结构体大小是不是一样?

附上:各数据类型的字节长度

运行以下代码,查看打印结果:

这个结果有没有在意料之中,以64位系统为例,struct1存取的字节数: double为8 、char为1、int为4、short为2,这就奇怪了,为什么读取内存不是按字节大小一个个读呢?这样就是8 + 1 + 4 + 2 = 15,这样看来不是更加省内存吗?通过查阅资料发现,处理器一般会以2字节,4字节,8字节,16字节甚至32字节为单位来存取内存。以这个思路重新计算一下,double以8字节存取,char以2字节存取,int以4字节存,short以2字节存,这样就是8 + 2 + 4 +2 = 16,并不是24,反而是struct2的大小。有没有可能计算机有一种策略,为了不在各种字节存取反复横跳,例如一会4字节存取,一会2字节存取,如果这样做会导致计算机开销巨大,是不是他就会定下一种策略,统一一个字节去存取,比如统一使用8字节存取,那么4个类型就是8 * 4 = 32,这里似乎又超过24,那么很可能的是计算机既做到了不用反复横跳的存取,又实现了内存的节省,究竟是怎么回事呢?

猜测1:结构体是不是将第一个成员变量作为首地址先存取了,然后后续的成员,再按谁最大,就以多少字节存取?那么来算一下:

double : 8   ,后面3个成员分别是 char int short , int占4个字节,所以后面以4字节读取,则通过计算得出

8 + 4 + 4 + 4 = 20 ,还是不对。

猜测2:第一个成员变量作为首地址先存取,后续成员的存取是不是和当前读取到的位置存在倍数关系,再进行一次计算

double   : 8       【0 ~  7】

char       : 1      【8】 有倍数关系

int          : 4            (9 , 10 , 11 ) 无倍数关系  【12】有倍数关系,从这开始读4位就是【12 13 14 15】

short     :  2         【16】有倍数关系 ,那就是【16,17】

竟然是奇数!好像还是不对,难道他的总大小也会有倍数关系吗?看结果 24,确实是和里面的任意一个成员都有倍数关系,那么我是不是可以合理猜测,当内存计算出的结果和里面的成员变量没有倍数关系时,他会自动补齐后续内存以达成倍数关系。从17 到 24之间 ,也只有24和所有成员变量有倍数关系,只要总大小和里面最大的成员成倍数关系,那么与其他的都是倍数关系,这也是为什么不是20,而是24.以这种思路计算struct2

double  : 8    【0 ~  7】

int         : 4     【8 9 10 11】

char      : 1     【12】

short     : 2      【 14  15 】

这里只读取到15,和 8 并不存在倍数关系,往后再读一位 那么就是 16

我们再写多几个结构体试试


计算struct 4:

 char      : 1    【0】

 char      : 1    【1】

 short     : 2    【 2  3 】

 int         : 4    【4 5 6 7】

最后补齐结果为 : 8 ,是4的倍数

计算struct 5

int           4      【0 1 2 3】

double    8      【8 9 10 11 12 13 14 15】

long       8       【16 17 18 19 20 21 22 23】

short      2       【24 25】

最后同样要补齐结果为: 32  , 是 8的倍数

打印验证:

继续往下研究,如果结构体里也有结构体,如下图,结构体3包含了结构体5

 我们依旧按照前面的方法计算前5个成员

double      8       [ 0  ~ 7]

int             4       [8 ~ 11]

char          1       [12]

short          2      [14 15]

int              4      [16 ~ 19]

那么问题来了,我们已知struct5是32,那他是从32开始,然后直接32 + 32等于64呢?还是需要从结构体5内部开始存呢?如果从结构体5内部的话,那么接着上面计算结构体5内部:

int           4    [20 ~ 23]

double    8   [24 ~ 31]

long        8   [32 ~ 39]

short       2   [40 41]

最后补齐倍数那就是8的倍数,往后存取就是48,

打印结果:

 结果真的打脸,既不是64也不是48,如果计算内部时,不是以第一个成员的倍数开始,而是以最大的成员的倍数开始,  结果又会怎样呢?接着计算:

int           4    [24 ~ 27]

double    8   [32 ~ 39]

long        8   [40 ~ 47]

short       2   [48 49]

最后补齐:56

 我的天,还真的是这样 ?为了进一步验证,我把struct5换成struct4,计算出的结果是32,大家可以尝试一下


结论:

关于结构体内存对齐,我得出几点原则:

1.从第一个成员开始,首地址为0,后续成员的首地址需要和他们的大小成倍数关系。

2. 最终大小如果和结构体里最大的成员无法成倍数关系,必须从当前地址继续往后偏移,直到地址和该成员成倍数关系,那么当前的地址就是结构体最终大小。

3.如果是结构体嵌套结构体,计算大小过程并不是单一的把结构体大小加上其他成员变量,而是以结构体中内部最大的成员来找到和该成员对应的地址,然后再根据 1,2点进行运算,最终得出结果。


探索收获:
为什么要进行结构体对齐?

从文章开头举的两个结构体例子中,我们可以知道,即使结构体里面的内容相同,但是大小确不是一样。通过探索结构体对齐,我明白在设计结构体中,一个好的结构体,往往能做到存取过程中,既节省空间,又能节省时间,比如例子中结构体2就比结构体1节省了8个字节,在文章上面计算struct1过程中,我们可以在计算中知道,结构体1,既要空出多余的空间,又要为了节省存取时间去补上了多余的空间,比如17到24这部分空间就是浪费了,而结构体2就没有这方面问题,从对比发现,似乎在设计结构体时,成员较小的尽量放到一起的话,会更省空间。虽然结构体内存对齐是为了用空间换取时间,但是我们作为开发者,也是可以通过巧妙的设计去帮助结构体节省空间。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容