数组传递会复制所有元素,性能随长度下降;切片仅复制少量元信息并共享底层数组,传递高效。

在Go语言中,数组和切片是两种常见的数据结构,虽然它们看起来相似,但在性能和使用场景上存在显著差异。理解这些差异对编写高效程序至关重要。
数组是值类型,拷贝成本高
Go中的数组是固定长度的值类型。这意味着当你将一个数组赋值给另一个变量,或作为参数传递给函数时,整个数组的内容会被复制一遍。
例如:
func process(arr [1024]int) {// 每次调用都会复制 1024 个 int
}
var a [1024]int
process(a) // 复制发生在这里
对于大数组,这种复制会带来明显的内存开销和性能损耗。即使你只读取数组内容,也无法避免复制。
立即学习“go语言免费学习笔记(深入)”;
切片是引用类型,共享底层数组
切片虽然由三个字段(指针、长度、容量)组成,本质是值传递,但它内部包含一个指向底层数组的指针。因此,多个切片可以共享同一块数据,传递时只需复制少量元信息。
示例:
func handler(s []int) {fmt.Println(len(s))
}
data := make([]int, 1000)
handler(data) // 只复制 slice header,不复制底层数组
这种设计使得切片在函数间传递非常高效,无论其长度多大,开销几乎不变。
值类型切片?不存在的误解
有人提到“值类型切片”,这其实是个误解。切片本身是引用语义的值类型:它按值传递结构体(指针+长度+容量),但其中的指针指向共享数据。因此,修改切片元素会影响原数据,而重新切分或扩容可能生成新底层数组。
对比:
- 数组传递:复制全部元素 → 开销随长度增长
- 切片传递:仅复制 header(约24字节)→ 固定小开销
实际性能建议
在性能敏感的场景中:
- 避免将大数组直接作为参数传递,应使用指针或转为切片
- 优先使用切片处理动态或较大规模数据
- 小数组(如 [4]byte、[16]byte)可接受值传递,因复制成本低
- 若需保护数据不被修改,应显式复制或使用只读接口
基本上就这些。关键在于理解:数组值传递 = 复制所有元素;切片传递 = 复制轻量结构并共享数据。合理选择能有效提升程序效率。











