Golang中传值与传指针调用有什么区别_Golang函数参数传递机制解析

Go语言中所有参数传递均为传值。传值时复制变量副本,函数内修改不影响原值,适用于基本类型和小结构体;传指针时复制地址副本,通过解引用可修改原数据,常用于大结构体或需修改原值场景。例如,modifyValue(x int) 修改形参不影响实参,而 modifyPointer(x *int) 通过指针修改使原值改变。结构体传递中,updateByName(u User) 无法更改原始对象,updateByPointer(u *User) 则可以。关键在于理解传递的是值本身还是指向值的地址:传值安全但可能低效,传指针高效但需防范并发与意外修改。

在Go语言中,函数参数传递始终是传值,也就是说,调用函数时会复制实参的值传递给形参。但根据传入的是普通变量还是指针对应的数据,行为表现不同,这常让人误解为“传值”和“传引用”。实际上Go中没有传引用机制,只有传值。区别在于你传的是值本身,还是指向值的指针。

传值:复制整个变量

当你将一个变量以值的方式传入函数,函数接收到的是该变量的一个副本。在函数内部对参数的修改不会影响原始变量。

适用类型:基本类型(int、string、bool等)和小结构体。

例如:

func modifyValue(x int) {
    x = 100
}
func main() {
    a := 10
    modifyValue(a)
    fmt.Println(a) // 输出:10,原值未变
}

这里 xa 的副本,修改 x 不会影响 a

传指针:复制指针地址

当你传入的是指针,虽然仍然是“传值”——复制的是指针的地址值,但由于这个副本指向同一块内存,函数可以通过指针修改原始数据。

适用场景:需要修改原数据、大结构体(避免复制开销)、共享数据。

示例:

func modifyPointer(x *int) {
    *x = 100
}
func main() {
    a := 10
    modifyPointer(&a)
    fmt.Println(a) // 输出:100,原值被修改
}

这里传递的是 &a,即 a 的地址。函数内通过 *x 解引用修改了原始值。

结构体的传值与传指针对比

结构体较大时,传值会带来显著的内存复制开销,而传指针只复制地址(通常8字节),效率更高。

type User struct {
    Name string
    Age  int
}

func updateByName(u User) {
    u.Name = "Updated"
}

func updateByPointer(u *User) {
    u.Name = "Updated"
}

调用 updateByName 不会改变原结构体;而 updateByPointer 可以直接修改原对象。

总结关键点

  • Go所有参数传递都是传值,不存在传引用。
  • 传值适合小对象,安全但可能有性能损耗。
  • 传指针可修改原数据,适用于大结构体或需状态变更的场景。
  • 指针传参虽高效,但需注意并发访问和意外修改问题。

基本上就这些。理解传的是“值”还是“指向值的地址”,就能清楚函数间数据交互的行为差异。不复杂但容易忽略细节。