指针可修改slice本身,如通过*[]T在函数中append;slice元素可为指针以避免复制;传指针才能改变slice头,截取时注意内存泄漏,必要时用copy隔离数据。

在Go语言中,指针与slice的交互非常常见,尤其是在需要修改slice内容或提高性能时。理解它们如何协作,能帮助写出更高效、更安全的代码。
指针指向slice本身
slice是引用类型,底层包含指向底层数组的指针、长度和容量。你可以定义一个指向slice的指(*[]T),这样可以在函数中修改slice本身(比如重新分配或截取)。
例如,想在一个函数中修改原始slice,就需要传入slice的指针:
- func modifySlice(s *[]int) {
- *s = append(*s, 100)
- }
- data := []int{1, 2, 3}
- modifySlice(&data)
- // data 现在是 [1 2 3 100]
如果不传指针,函数接收到的是slice的副本,对它的append操作不会影响原slice。
立即学习“go语言免费学习笔记(深入)”;
slice中存储指针元素
slice的元素可以是指针类型,如 []*Person,这在处理大型结构体时很实用,避免复制开销。
- type Person struct {
- Name string
- }
- people := []*Person{{Name: "Alice"}, {Name: "Bob"}}
- people[0].Name = "Alicia" // 直接修改指针指向的对象
这种方式共享数据,修改会影响所有引用该对象的地方,需注意并发安全。
函数间传递slice的注意事项
虽然slice本身是引用类型,但它的“头”(指针、len、cap)是值传递。如果函数需要改变slice的长度或让它指向新数组,必须传指针。
常见场景包括:
- 初始化nil slice:函数内用make创建,需通过指针赋值回传
- 大量append导致扩容:原slice无法感知新底层数组地址
- 重置或裁剪slice:如 s = s[:0],不传指针无效
避免常见陷阱
使用指针与slice交互时,容易出现内存泄漏或意外共享。
比如从大slice截取小slice,新slice仍指向原数组,导致原数据无法被GC。必要时可用 copy 创建完全独立的副本。
- small := make([]int, len(large[:3]))
- copy(small, large[:3])
当涉及指针slice时,确保理解每个指针的生命周期,避免悬挂指针或重复释放。
基本上就这些。掌握指针与slice的交互方式,能让你更好地控制数据共享与修改,写出更清晰的Go代码。










