首页 > 后端开发 > Golang > 正文

在 Go 中创建多维数组:数组的数组 vs. 切片的切片

花韻仙語
发布: 2025-10-08 10:33:21
原创
588人浏览过

在 go 中创建多维数组:数组的数组 vs. 切片的切片

本文将深入探讨在 Go 语言中创建多维数组的两种主要方法:数组的数组和切片的切片。我们将详细比较这两种方法的内部机制、内存使用情况、灵活性以及作为函数参数时的行为差异,并通过代码示例展示它们的特性和适用场景,帮助开发者根据实际需求做出最佳选择。

数组的数组

在 Go 语言中,我们可以创建一个数组,其元素本身也是数组,从而形成“数组的数组”,也称为多维数组。 这种方式在内存中是连续存储的,因此访问效率较高,尤其是在处理固定大小的矩阵或表格数据时。

package main

import "fmt"

func main() {
    // 创建一个 2x2 的整型数组的数组
    a := [2][2]int{{0, 1}, {2, 3}}

    // 遍历并打印数组元素及其内存地址
    fmt.Println("Array of Arrays")
    for i := 0; i < 2; i++ {
        for j := 0; j < 2; j++ {
            fmt.Printf("a[%d][%d] = %d at %p\n", i, j, a[i][j], &a[i][j])
        }
    }
}
登录后复制

这段代码创建了一个 [2][2]int 类型的数组 a,并使用嵌套循环遍历了它的所有元素,同时打印了每个元素的数值及其在内存中的地址。 由于是数组的数组,所以内存是连续分配的。

切片的切片

另一种创建多维结构的方法是使用“切片的切片”。 切片是 Go 语言中一种动态数组,它比数组更加灵活。 切片的切片允许每一行(或子切片)拥有不同的长度,因此非常适合处理不规则的数据结构。

package main

import "fmt"

func main() {
    // 创建一个 2x2 的整型切片的切片
    b := [][]int{{0, 1}, {2, 3}}

    // 遍历并打印切片元素及其内存地址
    fmt.Println("Slice of Slices")
    for i := 0; i < 2; i++ {
        for j := 0; j < 2; j++ {
            fmt.Printf("b[%d][%d] = %d at %p\n", i, j, b[i][j], &b[i][j])
        }
    }
}
登录后复制

这段代码创建了一个 [][]int 类型的切片 b,并同样遍历并打印了元素值和地址。与数组的数组不同,切片的切片中的每个子切片可能在内存中位于不同的位置。

内存使用比较

由于数组的数组在内存中是连续存储的,因此其内存占用通常比切片的切片更小。 切片的切片需要额外的空间来存储每个子切片的头部信息,并且子切片的数据可能分散在内存的不同位置。

以下是一个简单的对比示例,展示了创建大量数组的数组和切片的切片时的内存使用差异:

AssemblyAI
AssemblyAI

转录和理解语音的AI模型

AssemblyAI 65
查看详情 AssemblyAI
package main

import "fmt"
import "runtime"

func main() {
    // Array of Arrays
    var m runtime.MemStats
    runtime.GC()
    runtime.ReadMemStats(&m)
    alloc1 := m.Alloc

    a := [100000][3]int{}
    _ = a

    runtime.GC()
    runtime.ReadMemStats(&m)
    alloc2 := m.Alloc

    fmt.Printf("Array of Arrays uses %d bytes\n", alloc2-alloc1)

    // Slice of Slices
    runtime.GC()
    runtime.ReadMemStats(&m)
    alloc1 = m.Alloc

    b := make([][]int, 100000)
    for i := range b {
        b[i] = make([]int, 3)
    }
    _ = b

    runtime.GC()
    runtime.ReadMemStats(&m)
    alloc2 = m.Alloc

    fmt.Printf("Slice of Slices uses %d bytes\n", alloc2-alloc1)
}
登录后复制

在运行这段代码后,你会发现切片的切片占用的内存明显多于数组的数组。

作为函数参数的差异

数组和切片在作为函数参数时,表现出不同的行为。 数组是值类型,当数组作为参数传递给函数时,会创建一个数组的副本。 这意味着在函数内部对数组的修改不会影响原始数组。 切片是引用类型,当切片作为参数传递给函数时,传递的是切片的引用(或者说是切片头部的拷贝,但底层数组是共享的)。 因此,在函数内部对切片的修改会影响原始切片。

package main

import "fmt"

func f1(a [2][2]int) {
    fmt.Println("I'm a function modifying an array of arrays argument")
    a[0][0] = 100
}

func f2(b [][]int) {
    fmt.Println("I'm a function modifying an slice of slices argument")
    b[0][0] = 100
}

func main() {
    fmt.Println("Array of arrays")
    a := [2][2]int{{0, 1}, {2, 3}}
    fmt.Printf("Before %v\n", a)
    f1(a)
    fmt.Printf("After %v\n\n", a)

    fmt.Println("Slice of slices")
    b := [][]int{{0, 1}, {2, 3}}
    fmt.Printf("Before %v\n", b)
    f2(b)
    fmt.Printf("After %v\n", b)
}
登录后复制

运行结果:

Array of arrays
Before [[0 1] [2 3]]
I'm a function modifying an array of arrays argument
After [[0 1] [2 3]]

Slice of slices
Before [[0 1] [2 3]]
I'm a function modifying an slice of slices argument
After [[100 1] [2 3]]
登录后复制

可以看到,f1 函数修改了数组的副本,原始数组 a 保持不变。 而 f2 函数修改了切片的底层数组,因此原始切片 b 也被修改了。

总结

  • 数组的数组: 适用于固定大小的多维数据结构,内存效率高,但灵活性较差。作为函数参数时,传递的是数组的副本。
  • 切片的切片: 适用于大小可变的多维数据结构,灵活性高,但内存效率较低。作为函数参数时,传递的是切片的引用,对切片的修改会影响原始切片。

在选择使用哪种方法时,需要根据具体的应用场景权衡利弊。 如果需要处理固定大小的数据,并且对性能有较高要求,那么数组的数组是更好的选择。 如果需要处理大小可变的数据,或者需要在函数内部修改多维数据结构,那么切片的切片是更合适的选择。一般来说,对于一维数据,切片通常优于数组。

以上就是在 Go 中创建多维数组:数组的数组 vs. 切片的切片的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号