数组是固定长度的数据结构,适合数据量小、长度固定的场景,如存储颜色rgb值或作为切片的底层存储;切片是对数组的封装,提供动态扩容、高效传参和子序列处理能力,适用于大多数集合数据操作场景。数组是值类型,声明时需指定长度且不可变,而切片是引用类型,底层指向数组并包含指针、长度和容量,支持运行时动态增长。切片在扩容时,若容量小于1024则翻倍,超过则增加约25%,以平衡性能与内存使用。选择数组的典型情况包括:数据长度固定、性能要求极高或作为哈希键使用;选择切片的情况包括:数据长度不确定、需要截取子集、拼接或传递给函数,以及大多数业务逻辑中的集合操作。

在Go语言中,数组和切片是存储和操作数据的基础结构,但它们的底层实现和使用场景差异很大。简单来说:数组是固定长度的数据结构,而切片是对数组的封装,提供了更灵活的操作方式。

如果你只是想临时存几个元素,用哪个都行;但一旦涉及动态扩容、子序列处理或者函数传参,切片就成了首选。

数组的底层结构与适用场景
数组在Go中是值类型,声明时必须指定长度,且不能改变大小。比如:
立即学习“go语言免费学习笔记(深入)”;
var arr [5]int
这会分配一块连续的内存空间,存储5个整型值。数组的底层就是一段固定的内存块,访问速度快,适合数据量小、长度固定的场景。

常见用途包括:
- 存储固定长度的数据,如颜色RGB值
[3]byte - 作为切片的底层存储结构
- 需要高性能、无额外开销的场合
但它的缺点也很明显:扩容困难、传参效率低(复制整个数组)。
切片的结构与灵活性优势
切片在Go中是一个引用类型,它并不直接保存数据,而是指向一个底层数组。其结构大致如下:
struct {
array unsafe.Pointer // 指向底层数组
len int // 当前长度
cap int // 底层数组容量
}正因为这种设计,切片可以在运行时动态增长,例如:
s := []int{1, 2, 3}
s = append(s, 4)这里如果当前底层数组容量不足,Go运行时会自动分配一个更大的数组,并将原数据复制过去。
使用切片的好处包括:
- 支持动态扩容
- 函数传参高效(只传指针、长度和容量)
- 可以方便地截取子序列
s[1:3]
适用于大多数需要处理集合数据的场景,比如读取HTTP请求参数、解析JSON数组等。
切片扩容机制简析
切片的扩容机制是其性能关键之一。当你不断调用 append 超出当前容量时,Go会根据一定策略重新分配内存。
一般情况下:
- 如果当前容量小于1024,每次扩容为原来的2倍
- 如果超过1024,每次增加25%左右
举个例子:
s := make([]int, 0, 4)
for i := 0; i < 10; i++ {
s = append(s, i)
}初始容量是4,当插入第5个元素时,就会触发扩容,变成8;再继续插入到9个时,可能扩容到16。
这个机制避免了频繁申请内存,也保证了性能稳定。
如何选择数组还是切片?
-
用数组的情况:
- 数据长度固定,不会变化
- 对性能要求极高,不想引入额外开销
- 作为哈希键使用(因为数组可以比较,切片不行)
-
用切片的情况:
- 数据长度不确定或经常变化
- 需要截取子集、拼接、传递给函数
- 大多数业务逻辑中的集合操作
其实大多数时候你都不需要考虑数组,除非你知道为什么用它更好。
基本上就这些区别了。










