切片传参是值传递,传递的是包含指针、长度和容量的切片头副本。由于副本中的指针仍指向原底层数组,因此修改元素会直接影响原切片;但扩容或重新赋值仅改变副本的指针或长度,不会影响原切片。

Go语言中,切片(slice)传参是值传递,但传递的是切片头的副本,不是指针类型。
切片的本质是结构体
切片在底层是一个包含三个字段的结构体:
- Pointer:指向底层数组的指针
- len:当前切片的长度
- cap:当前切片的容量
当把切片作为参数传递时,Go会复制这个结构体(即“切片头”),但不会复制它指向的底层数组。因此,函数内部和外部的切片共享同一块底层数组。
为什么修改元素会影响原切片?
因为传入的是切片头的副本,其中的指针仍指向原来的底层数组。所以通过索引修改元素时,实际上是修改了共享的数组数据。
例如:
func modify(s []int) {s[0] = 99
}
func main() {
arr := []int{1, 2, 3}
modify(arr)
fmt.Println(arr) // 输出 [99 2 3]
}
扩容或重新赋值不影响原切片
如果在函数内对切片进行 append 导致扩容,或者重新赋值,只会改变副本的指针、长度或容量,不会影响原切片。
例如:
func reassign(s []int) {s = append(s, 4, 5, 6) // 可能触发扩容
}
func main() {
arr := []int{1, 2, 3}
reassign(arr)
fmt.Println(arr) // 仍输出 [1 2 3]
} 基本上就这些。切片传参是值传递,但由于包含指针,部分操作会影响原数据,需注意区分场景。










