一文探索Go语言中的内存对齐

在 Go 语言中,内存对齐是一个经常被忽略但非常重要的概念。理解内存对齐不仅可以帮助我们写出更高效的代码,还能避免一些潜在的性能陷阱。

在这篇文章中,我们将通过一个简单的例子来探讨 Go 语言中的内存对齐机制,以及为什么相似的结构体在内存中会占用不同的大小。

示例代码

我们先来看一段代码:

package memory_alignment

import (
	"fmt"
	"unsafe"
)

type A struct {
	a int8
	b int8
	c int32
	d string
	e string
}

type B struct {
	a int8
	e string
	c int32
	b int8
	d string
}

func Run() {
	var a A
	var b B
	fmt.Printf("a size: %v \n", unsafe.Sizeof(a))
	fmt.Printf("b size: %v \n", unsafe.Sizeof(b))
	// a size: 40
	// b size: 48
}

在这个例子中,我们定义了两个结构体 AB。它们的字段基本相同,只是排列顺序不同。然后,我们使用 unsafe.Sizeof 来查看这两个结构体在内存中的大小。

结果却令人惊讶:结构体 A 的大小是 40 字节,而结构体 B 的大小是 48 字节。为什么会出现这样的差异呢?这就是我们今天要讨论的内存对齐的作用。

内存对齐概念

内存对齐是指编译器为了优化内存访问速度,而对数据在内存中的位置进行调整的一种策略。不同类型的数据在内存中的对齐要求不同,例如:

  • int8 类型的变量通常对齐到 1 字节边界。
  • int32 类型的变量通常对齐到 4 字节边界。
  • 指针(如 string)通常对齐到 8 字节边界。

为了满足这些对齐要求,编译器可能会在结构体的字段之间插入一些“填充”字节,从而确保每个字段都能正确对齐。

结构体内存布局解析

让我们深入分析一下 AB 两个结构体的内存布局,看看编译器是如何为它们分配内存的。

结构体 A 的内存布局

| a (int8) | b (int8) | padding (2 bytes) | c (int32) | d (string, 8 bytes) | e (string, 8 bytes) |
  • abint8 类型,各占 1 字节。
  • cint32 类型,需要 4 字节对齐,b 后面会有 2 个填充字节。
  • destring 类型,各占 8 字节。

总大小为:1 + 1 + 2 + 4 + 8 + 8 = 24 字节。

结构体 B 的内存布局

| a (int8) | padding (7 bytes) | e (string, 8 bytes) | c (int32) | padding (4 bytes) | b (int8) | padding (3 bytes) | d (string, 8 bytes) |
  • aint8 类型,占 1 字节,后面有 7 个填充字节,以便 e 能够对齐到 8 字节边界。
  • cint32 类型,需要 4 字节对齐,因此在 c 后面没有填充。
  • bint8 类型,需要填充 3 个字节来对齐到 d 的 8 字节边界。

总大小为:1 + 7 + 8 + 4 + 4 + 1 + 3 + 8 = 36 字节。

请注意,Go 编译器可能会将 de 视为 8 字节对齐类型(取决于系统和编译器的实现),因此总大小可能是 48 字节。

如何优化结构体内存布局

为了减少结构体的内存占用,我们可以按照字段的对齐要求来重新排列字段。例如:

先声明大的字段(如 stringint32),然后是小的字段(如 int8),可以减少内存中的填充字节。

我们可以将 B 结构体改成以下形式:

type OptimizedB struct {
    e string
    d string
    c int32
    a int8
    b int8
}

这样可以减少内存填充,从而优化内存占用。

总结

内存对齐是编译器优化内存访问速度的一个重要策略。虽然它对大多数应用程序的影响可能较小,但在高性能场景或内存受限的环境中,理解并优化内存对齐可能会带来显著的性能提升。

在 Go 语言中,了解结构体的内存对齐规则,合理排列结构体字段顺序,不仅可以提高程序的性能,还能减少内存的浪费。这是一种简单而有效的优化手段,希望大家在以后的编程实践中能够灵活运用。

到此这篇关于一文探索Go语言中的内存对齐的文章就介绍到这了,更多相关Go内存对齐内容请搜索恩蓝小号以前的文章或继续浏览下面的相关文章希望大家以后多多支持恩蓝小号!

原创文章,作者:EYJYV,如若转载,请注明出处:http://www.wangzhanshi.com/n/5563.html

(0)
EYJYV的头像EYJYV
上一篇 2024年12月17日 19:27:41
下一篇 2024年12月17日 19:27:43

相关推荐

  • Go中log包异或组合配置妙用详解

    log 中的这种用法,你一定见过: log.SetFlags(log.Ldate | log.Ltime | log.Llongfile) 没见过的,自我反省下(逃 在 Go 语言…

    Golang 2024年12月17日
  • golang gorm更新日志执行SQL示例详解

    1. 更新日志 1.1. v1.0 1.1.1. 破坏性变更 gorm.Open返回类型为*gorm.DB而不是gorm.DB 更新只会更新更改的字段 大多数应用程序不会受到影响,…

    Golang 2024年12月29日
  • 重学Go语言之基础数据类型详解

    前言 Go语言有非常强大的数据类型系统,其支持的数据类型大体上可分为四类:基础数据类型、引用数据类型、接口类型、复合类型。 基础数据类型有: 布尔型(bool) 整型(int) 浮…

    Golang 2024年12月29日
  • go中结构体切片的实现示例

    在 Go 语言中,结构体切片是一种非常常用的数据结构,它结合了结构体和切片的特点,可以方便地存储和操作多个结构体实例。以下是关于 Go 结构体切片的详细介绍: 1. 结构体切片的定…

    Golang 2024年12月17日
  • golang实现ip访问限制及提交次数

    在 Web 应用中,通常会需要对 IP 访问进行限制以及控制提交次数,以防止恶意攻击(例如暴力破解、DoS攻击、API滥用等)。为了实现这一功能,我们可以结合 Golang 的特性…

    Golang 2024年12月17日
  • 在Go中动态替换SQL查询中的日期参数的完整步骤

    完整指南:在Go中动态替换SQL查询中的日期参数 在处理数据库查询时,经常需要根据不同的输入条件动态地构造SQL语句。尤其是在涉及日期范围的查询中,能够根据实际需求调整查询的起始和…

    Golang 2024年12月17日
  • 如何使用Go检测用户本地是否安装chrome

    前言 起因,是因为我之前做过用 go 直接开启浏览器服务,然后在昨天遇到了一位大佬的业务,大佬使用 python 做了一个脚本,并打包成 exe 文件,让我帮其测试,但是在我电脑上…

    Golang 2024年12月17日
  • Golang动态数组的实现示例

    什么是动态数组 动态数组(Dynamic Array)是一种在需要时能够自动改变其大小的数组。与静态数组(Static Array)不同,静态数组的大小在声明时就已确定,且之后不能…

    2024年12月17日
  • ubuntu安装golang并设置goproxy的方法步骤

    在Ubuntu上安装Go语言(Golang)通常有几种方法,以下是一些常见的安装步骤: 方法一:使用包管理器安装 更新包列表: sudo apt update 安装Go: sudo…

    Golang 2024年12月17日
  • Go实现将任何网页转化为PDF

    在许多应用场景中,可能需要将网页内容转化为 PDF 格式,比如保存网页内容、生成报告、或者创建网站截图。使用 Go 编程语言,结合一些现有的库,可以非常方便地实现这一功能。本文将带…

    Golang 2024年12月17日

发表回复

登录后才能评论