Go切片是基于数组的动态集合,含指针、len、cap三属性;append实现自动扩容,截取语法控制共享与隔离;传参为结构体副本,修改元素影响原数组,清空推荐s=s[:0]。

Go语言的切片(slice)是管理动态数据集合的核心工具,它底层基于数组,但能自动扩容、灵活截取,用起来比数组更实用。
理解切片的三个关键属性:指针、长度和容量
每个切片包含三个字段:指向底层数组的指针、当前元素个数(len)、底层数组从切片起始位置开始可用的最大元素数(cap)。比如:
s := make([]int, 3, 5) // len=3, cap=5,底层数组有5个int,但只“暴露”前3个
理解 len 和 cap 的区别,是掌握扩容和安全截取的前提——截取不会改变底层数组,但可能共享内存;扩容则可能触发新数组分配。
立即学习“go语言免费学习笔记(深入)”;
动态扩容:append 是主力,但要注意底层数组复用
使用 append() 向切片末尾添加元素时,如果 cap 足够,就在原数组上追加;否则 Go 自动分配更大底层数组(通常是翻倍增长),并复制原有数据。
- 多次 append 小量数据时,不必预先计算容量,Go 会高效处理
- 若已知最终大小(如读取文件行数),用 make([]T, 0, estimatedCap) 预分配可避免多次复制
- 注意:append 返回新切片,原变量不自动更新,需显式赋值:s = append(s, x)
安全截取切片:用 [:] 语法,但别越界或误共享
截取语法 s[i:j:k](k 可选)控制新切片的 len 和 cap:
- s[i:j] → 新 len = j−i,cap = 原 cap − i
- s[i:j:k] → 新 len = j−i,cap = k−i(限制后续 append 不影响原数据)
- 所有索引必须满足 0 ≤ i ≤ j ≤ k ≤ cap,否则 panic
- 若需完全隔离(避免修改影响原切片),用 copy(dst, src) 复制一份独立副本
常见陷阱与实用建议
切片操作看似简单,但容易踩坑:
- 循环中直接 append 到同一个切片?没问题;但若在循环内截取并保存多个子切片,它们可能共用底层数组——后续修改会相互干扰
- 函数传参时传切片,实际传的是包含指针、len、cap 的结构体副本,所以函数内 append 不影响调用方,但修改元素会影响(因指针指向同一数组)
- 清空切片推荐写法:s = s[:0](保留底层数组,高效复用);不要用 nil 赋值,除非真想释放引用










