Go 中函数参数传递:指针与值拷贝的本质区别

1次阅读

Go 中函数参数传递:指针与值拷贝的本质区别

go 函数参数始终按值传递;传指针本质是传递地址值的拷贝,而传基本类型则是传递数据本身的拷贝——二者语义和效果截然不同。

go 函数参数始终按值传递;传指针本质是传递地址值的拷贝,而传基本类型则是传递数据本身的拷贝——二者语义和效果截然不同。

在 Go 中,不存在传统意义上的“引用传递,所有函数参数都是按值传递(pass by value)。关键在于:你传递的“值”本身是什么类型。若传递的是 *int,那么被复制的是该指针变量所存储的内存地址;若传递的是 int,则被复制的是整数值本身。这一根本差异直接决定了函数内部能否修改调用方的原始变量。

✅ 正确做法:显式传递指针,实现“修改原值”

以下代码通过 &x 将变量 x 的地址传入函数,add1 接收 *int 类型参数,解引用后直接操作原始内存:

func add1(a *int) int {     *a = *a + 1 // 修改 a 所指向的内存位置中的值(即 x 本身)     return *a }  func main() {     x := 3     fmt.Println("x =", x)        // x = 3     x1 := add1(&x)               // 传地址     fmt.Println("x+1 =", x1)     // x+1 = 4     fmt.Println("x =", x)        // x = 4 ← 原变量已被修改 }

✅ 成功修改了 x,因为 &x 提供了通往 x 内存空间的“钥匙”,*a 即是对该空间的直接写入。

❌ 常见误解:在函数内取局部变量地址无法影响外部

如下写法看似“用了指针”,实则无效:

func add1(a int) int {     p := &a   // &a 取的是形参 a(局部拷贝)的地址!     *p = *p + 1     return *p }  func main() {     x := 3     fmt.Println("x =", x)        // x = 3     x1 := add1(x)                // 传值:a 是 x 的独立拷贝     fmt.Println("x+1 =", x1)     // x+1 = 4     fmt.Println("x =", x)        // x = 3 ← 原变量未变! }

⚠️ 注意:a 是 x 的一份独立副本,生命周期仅限于 add1 函数作用域。&a 得到的是这个副本的地址,*p 修改的只是副本 a,对调用方的 x 完全无影响。

? 核心总结

场景 传递内容 是否修改原始变量 原因
add1(&x) x 的内存地址(如 0xc0000140a0) ✅ 是 函数通过该地址访问并修改 x 所在内存
add1(x) 整数值 3 ❌ 否 函数内 a 是新分配的局部变量,与 x 内存无关

? 最佳实践提示

  • 若需函数修改调用方变量,请明确使用指针类型参数,并在调用时加 &;
  • 避免在值参数内部取地址并误以为能影响外部——这仅操作局部副本;
  • Go 的设计哲学是“显式优于隐式”:是否可变,由类型签名清晰表达(int vs *int),而非依赖语言机制“自动引用”。

理解这一机制,是写出安全、高效、符合 Go 惯例代码的基础。

text=ZqhQzanResources