
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 惯例代码的基础。










