登录
首页 >  Golang >  Go教程

一文带你搞懂Golang结构体内存布局

来源:脚本之家

时间:2022-12-22 19:11:26 125浏览 收藏

本篇文章主要是结合我之前面试的各种经历和实战开发中遇到的问题解决经验整理的,希望这篇《一文带你搞懂Golang结构体内存布局》对你有很大帮助!欢迎收藏,分享给更多的需要的朋友学习~

结构体内存布局

结构体大小

结构体实际上就是由各种类型的数据组合而成的一种符合数据类型,一个结构体变量的大小是由结构体中的字段决定。结构体和它所包含的数据在内存中是以连续块的形式存在的。我们可以借助unsafe.Sizeof方法,来获取:

package main

import (
	"fmt"
	"unsafe"
)

type Test struct {
	T1 int8 // 1
	T2 int8 // 1
	T3 int8 // 1
}
func main() {
    var t Test
	fmt.Println(unsafe.Sizeof(t)) //3 bytes

}

内存对齐

不同类型的变量占用内存大小是不一样的,但是cpu每次读取的内存长度是固定的(比如cpu是64位的,一次可以从内存中读取64位的数据,即8个字节),为了cpu能高效的读写数据,编译器会把各种类型的数据放在合适的地址,而不是顺序的一个接一个的排放,并占用合适的长度,这就是内存对齐。每种类型的对齐值就是它的对齐边界。

示例:

package main

import (
	"fmt"
	"unsafe"
)
type Test struct {
	a int8 // 1
	b int64 // 8
	c int32 // 4
}
type Test2 struct {
	a int8 // 1
	b int32 // 4
	c int64 // 8
}
func main() {
    var t Test
	fmt.Println(unsafe.Sizeof(t)) // 24
	var t2 Test2
	fmt.Println(unsafe.Sizeof(t2))// 16
}

通过上面示例,我们可以看到两个结构体中3个字段类型相同,当排列顺序发生变化时,总的内存大小也会发生变化。下面我们来一起分析一下:

如果没有内存对齐,那结构体各个字段在内存中是紧密排列的,如t1内存布局示意图如下:

因为b这个字段需要8个字节,所以会有一个字节的数据排列到第2个字中。如果程序想要读取b字段的数据,那么CPU需要两次读取才能获取到完整的数据,这样就会影响了程序的性能。

所以,为了能让CPU减少一次获取的时间,Go编译器会帮你把struct结构体做数据的对齐,以便CPU可以一次将该数据从内存中读取出来。重新排列后内存布局结构示意图如下:

其中有13个字节是真正存储数据的,而灰色的11个字节则是为了对齐而填充上的,不存储任何数据,所以才会比没有对齐排列时多出11个字节。

虽然通过对齐填充的方式提高了CPU读写数据的效率,但是这些填充内存确实有点浪费空间,那有没有办法既可以既可以做到内存对齐保证CPU读写效率又能减少浪费内存空间呢?

那就是调整struct字段的顺序,我们在来看一下t2结构体的字段内存布局结构示意图如下:

这样重新排列后,只占了16个字节,比上面那种方式少了8个字节。由此可知,对结构体字段的重新排列会让结构体更节省内。

总结

本篇文章我们一起学习了Go 语言中的内存对齐,主要内容如下:

  • 结构体是占用一块连续的内存,一个结构体变量的大小是由结构体中的字段决定。
  • unsafe.Sizeof(x) 返回了变量x的内存占用大小。
  • 两个结构体,即使包含变量类型的数量相同,但是位置不同,占用的内存大小也不同,由此引出了内存对齐。
  • 对结构体字段的重新排列会让结构体更节省内。

文中关于golang的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《一文带你搞懂Golang结构体内存布局》文章吧,也可关注golang学习网公众号了解相关技术文章。

声明:本文转载于:脚本之家 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>
评论列表