Go中所有参数传递均为值拷贝,拷贝内容可能是数据本身或指针地址。1. 值拷贝:传基本类型、结构体时复制副本,函数内修改不影响原变量;2. 指针传递:拷贝的是地址,通过指针可修改原始数据;3. 特殊类型如slice、map、channel底层含指针,传递时拷贝的是指向底层数组的指针结构,故能间接修改数据,但重新赋值不影响外部;4. 选择依据:是否需修改原数据、数据大小、方法集一致性与nil安全性。核心是理解拷贝的是值还是指针。

在Go语言中,理解值拷贝与指针传递对编写高效、正确的程序至关重要。很多人刚接触Go时会对函数参数的传递方式产生误解,认为像切片或map是“引用传递”,其实Go中所有参数传递都是值拷贝,只是有时拷贝的是指针而已。
值拷贝:传递的是数据的副本
当一个变量以值的方式传入函数时,Go会创建该变量的一个副本。函数内部对参数的修改不会影响原始变量。
基本类型如 int、float、bool、string 以及结构体默认都是值拷贝:
type Person struct {
Name string
Age int
}
func modifyPerson(p Person) {
p.Name = "Alice"
p.Age = 30
}
func main() {
person := Person{Name: "Bob", Age: 25}
modifyPerson(person)
fmt.Println(person) // 输出 {Bob 25},原值未变
}
上面的例子中,modifyPerson 接收的是 person 的一份拷贝,内部修改不影响原始结构体。
立即学习“go语言免费学习笔记(深入)”;
指针传递:拷贝的是地址
要让函数能修改原始变量,需要传递变量的地址(即指针)。虽然这仍是“值拷贝”——拷贝的是指针的值,但由于指针指向同一块内存,因此可以通过它修改原数据。
func modifyPersonPtr(p *Person) {
p.Name = "Alice"
p.Age = 30
}
func main() {
person := Person{Name: "Bob", Age: 25}
modifyPersonPtr(&person)
fmt.Println(person) // 输出 {Alice 30},原值被修改
}
这里传递的是 &person,即 person 的地址。函数接收 *Person 类型,通过指针访问并修改原始结构体。
特殊类型的行为:slice、map、channel
虽然 slice、map 和 channel 不是基本类型,但它们底层由指针封装。当你传递这些类型时,拷贝的是包含指针的结构体(如 slice header),所以它们能“间接”修改底层数组或数据结构。
func modifySlice(s []int) {
s[0] = 999
}
func main() {
data := []int{1, 2, 3}
modifySlice(data)
fmt.Println(data) // 输出 [999 2 3]
}
尽管是值拷贝,但拷贝的 slice 仍指向相同的底层数组,因此修改生效。但若在函数内重新分配 slice,外部不会感知:
func reassignSlice(s []int) {
s = append(s, 4, 5, 6) // 可能触发扩容,指向新数组
}
// 调用后 data 长度仍为3,不受影响
如何选择:值还是指针?
决定使用值还是指针作为参数类型,主要考虑以下几点:
- 是否需要修改原数据:需要修改就用指针
- 数据大小:大结构体拷贝成本高,建议传指针
- 一致性:如果方法集中有指针接收者,其他方法也建议用指针,避免混淆
- nil 安全性:值接收者不会因 nil 指针而 panic,指针接收者需注意判空
基本上就这些。记住一句话:Go只有值拷贝,指针传递只是拷贝了地址。关键在于理解拷贝的内容是什么——是数据本身,还是指向数据的指针。










