Go函数参数是值传递,但可通过传指针修改原变量或避免大对象拷贝;需用&取地址调用,注意nil检查和生命周期;指针方法要求接收者可寻址。
在 Go 中,函数参数是值传递的,但你可以显式地传递指针,从而让函数修改原始变量。关键不是“Go 支持引用传递”,而是“你传的是指向变量的地址,函数通过该地址读写原内存”。理解这点,就能正确使用指针参数。
主要出于两个目的:修改调用方的变量,或避免复制大对象(如结构体、切片底层数组等)。
json.Unmarshal 第二个参数必须是 *T)函数签名中明确写出指针类型,调用时用取地址符 & 传入变量地址。
例如:
func doubleValue(x *int) {
*x = *x * 2
}
a := 10
doubleValue(&a) // 传 &a,不是 a
fmt.Println(a) // 输出 20
注意:&a 的类型是 *int,必须和函数参数类型严格匹配;不能对字面量或表达式取地址(如 &(x+1) 是非法的)。
指针操作容易出错,尤其涉及 nil 和生命周期问题。
panic: runtime error: invalid memory address or nil pointer dereference),调用前应检查:if x != nil { *x = ... }
*(除非你要替换整个 slice header)如果某个结构体方法需要修改接收者字段,必须定义为指针接收者(func (s *MyStruct) Modify() {...})。此时调用该方法的对象也建议用指针传入,否则会复制一份再调用,修改无效。
例如:
type Counter struct{ n int }
func (c *Counter) Inc() { c.n++ }
c := Counter{0}
c.Inc() // OK:c 是可寻址的,编译器自动取 &c
c2 := Counter{0}
ptr := &c2
ptr.Inc() // 明确用指针调用
但如果写 foo(Counter{}) 再在内部调用 Inc(),就会报错:“cannot call pointer method on ...”。