数组是值类型,传参时整块复制;切片传的是含指针、len、cap的结构体,共享底层数组。[3]int与[4]int类型不同,而[]int无长度限制,可接收任意长度切片。
当你把一个 [5]int 传给函数,Go 会把这 5 个整数完整拷贝一份——哪怕只读,也开销不小;而传 []int 时,实际只拷贝一个含三个字段的结构体:array(指针)、len、cap。这意味着切片修改可能影响原底层数组,但结构体本身是值拷贝。
s2 = s1 后修改 s2 不会影响 s1 → 实际上它们共享底层数组,改元素会互相可见append([]int(nil), s1...) 或 copy 到新切片[3]int 和 [4]int 是完全不同的类型,不能赋值、不能比较、不能混用;而 []int 就是 []int,不管它背后是 3 个还是 300 个元素。
cannot use arr (variable of type [5]int) as [3]in
t value in assignment
func f(s []int) 可接收任意长度切片;若写 func f(a [5]int),只能传严格匹配的数组var a [3]int 的零值是 [3]int{0,0,0};var s []int 的零值是 nil(注意:len(s) 和 cap(s) 都为 0,但 s == nil 为 true)slice 没有自己存数据,它只是底层数组的一段窗口。append 超过 cap 时,Go 会分配新数组、复制旧数据、更新指针——这个过程不可见,但代价真实存在。
s := make([]int, 0, 4); s = append(s, 1,2,3,4,5) → 第 5 次 append 触发扩容,原底层数组被丢弃,后续所有基于旧 s 的切片(如 s[1:])仍指向老内存,但内容已不再更新cap(s),不是 len(s);扩容策略通常是翻倍(小容量)或加固定值(大容量),具体看 runtime 源码make([]int, 0, 100)
绝大多数业务代码里,你几乎不需要显式声明数组。但以下情况例外:
type IP [4]byte),此时数组长度是类型一部分,能保证内存布局稳定[16]byte 常用于临时缓冲),避免堆分配和 GC 压力[]byte 不行):m := map[[4]byte]int{[4]byte{1,2,3,4}: 1}
C.array 对应 Go 中的 [N]C.type)package main
import "fmt"
func main() {
// 数组:类型含长度,传参即拷贝
a := [3]int{1, 2, 3}
modifyArray(a) // a 不变
fmt.Println(a) // [1 2 3]
// 切片:传结构体,改元素影响底层数组
s := []int{1, 2, 3}
modifySlice(s) // s[0] 变成 999
fmt.Println(s) // [999 2 3]
}
func modifyArray(a [3]int) { a[0] = 999 }
func modifySlice(s []int) { s[0] = 999 }
底层数组是否被共享、扩容是否发生、类型能否匹配——这些不是“理论区别”,而是每次写 append、传参、赋值时实实在在影响行为的点。别依赖直觉,该打日志打日志,该查 cap 查 cap。