贝利信息

Golang数组与slice有什么区别_Array与Slice差异说明

日期:2026-01-18 00:00 / 作者:P粉602998670
数组是值类型,传参时整块复制;切片传的是含指针、len、cap的结构体,共享底层数组。[3]int与[4]int类型不同,而[]int无长度限制,可接收任意长度切片。

数组是值类型,传参时整块复制,切片传的是“视图结构体”

当你把一个 [5]int 传给函数,Go 会把这 5 个整数完整拷贝一份——哪怕只读,也开销不小;而传 []int 时,实际只拷贝一个含三个字段的结构体:array(指针)、lencap。这意味着切片修改可能影响原底层数组,但结构体本身是值拷贝。

长度是否属于类型?这是类型系统里最硬的分水岭

[3]int[4]int 是完全不同的类型,不能赋值、不能比较、不能混用;而 []int 就是 []int,不管它背后是 3 个还是 300 个元素。

扩容机制让 slice 看似“无限”,但底层数组其实很诚实

slice 没有自己存数据,它只是底层数组的一段窗口。append 超过 cap 时,Go 会分配新数组、复制旧数据、更新指针——这个过程不可见,但代价真实存在。

何时非用数组不可?真不多,但有且仅有的几个场景

绝大多数业务代码里,你几乎不需要显式声明数组。但以下情况例外:

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、传参、赋值时实实在在影响行为的点。别依赖直觉,该打日志打日志,该查 capcap