【Golang】内存对齐

欢迎关注微信公众号:全栈工厂

1. 先看一个问题

请思考30秒想想以下代码输出的内容是多少?

package main

import (
    "fmt"
    "unsafe"
)

type S1 struct {
    A byte
    B int64
    C byte
}
type S2 struct {
    A byte
    C byte
    B int64
}
func main() {
    s1 := S1{}
    fmt.Printf("S1.A size: %d\n", unsafe.Sizeof(s1.A))
    fmt.Printf("S1.B size: %d\n", unsafe.Sizeof(s1.B))
    fmt.Printf("S1.C size: %d\n", unsafe.Sizeof(s1.C))
    fmt.Printf("S1   size: %d\n\n", unsafe.Sizeof(s1))

    s2 := S2{}
    fmt.Printf("S2.A size: %d\n", unsafe.Sizeof(s2.A))
    fmt.Printf("S2.B size: %d\n", unsafe.Sizeof(s2.B))
    fmt.Printf("S2.C size: %d\n", unsafe.Sizeof(s2.C))
    fmt.Printf("S2   size: %d\n", unsafe.Sizeof(s2))
}

执行后代码输出:

S1.A size: 1
S1.B size: 8
S1.C size: 1
S1   size: 24

S2.A size: 1
S2.B size: 8
S2.C size: 1
S2   size: 16

你或许会疑问,为什么S1和S2所占存储空间不仅不是A、B、C总和10而且S1和S2所占空间也各不一样?问题的关键就在于"内存对齐"。

2. 什么是内存对齐?

在计算机中,CPU通过特定的指令从内存中读取数据,由于CPU访问内存已得到数据的时间要比执行指令花费的时间多得多,因此在CPU内部提供了一些通用寄存器用来暂存从内存中加载到的数据,CPU一次读取的数据量为一个字,字的位数我们称之为字长,因此字长的大小直接决定了CPU一次能处理的数据量的大小;一般情况下,字长越大,CPU性能越高,我们熟知的64位计算机就表示CPU一次能处理的数据量为64位即8字节。
因此,所谓内存对齐就是计算机将内存中的数据按照一个字的长度(即:CPU数据处理单位)进行对齐,保证CPU以高效的方式精确读取到所需要的数据。

3. 为什么要内存对齐?

3.1 提高性能

如果CPU不按照块(字)去读取数据,而是按照字节去读取数据,那么CPU读取一个int64值就需要读取8次,效率很低,所以最终CPU被设计一次读取一个字长的数据,内存的基本存储结构如下:


如果不进行内存对齐的话,一个int64在内存中的存储位置很可能像下面这样:


image.png

这样的话,CPU就需要读取两次才能完成这个int64值的读取操作,而进行内存对齐后,保证每个变量的值都存储在整数倍机器字长的内存地址上,最小化CPU内存读取次数,提高效率。

3.2 避免出错

不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常,出现错误。

4. 问题解答

了解内存对齐后,我们再看文章开始我们说道的那题,在我的64位计算机环境中,S1和S2在内存中的大致位置如下图所示:


所以,最后S1所占的内存空间为24字节,S2所占空间为16字节。

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

推荐阅读更多精彩内容

  • 什么是内存对齐 此时的 打印值为:16,一个64长度int占8字节,两个就是16。 此时的 打印值为:16,4+8...
    Stevennnmmm阅读 498评论 0 0
  • 一、结构体内存对齐 1.1 结构体内存对齐三大原则 数据成员对⻬规则结构体(struct)或联合体(union)的...
    HotPotCat阅读 468评论 1 3
  • 如何得到一个对象所占内存大小? 内存对齐: 为何会有内存对齐?1.并不是所有硬件平台都能访问任意地址上的任意数据。...
    郭老汉阅读 1,949评论 2 4
  • 先看一个结构体 对于这两个结构体,都有a、b、c三个定义完全一样的字段,只是在定义结构体的时候字段顺序不一样而已,...
    cfanbo阅读 88评论 0 0
  • 我是黑夜里大雨纷飞的人啊 1 “又到一年六月,有人笑有人哭,有人欢乐有人忧愁,有人惊喜有人失落,有的觉得收获满满有...
    陌忘宇阅读 8,535评论 28 53