go 中方法接收者是否加 `*` 决定了该方法是作用于原结构体实例还是其副本:指针接收者可修改原数据并共享状态,值接收者操作的是独立拷贝,对原值无影响。
在 Go 中,方法接收者类型(T 或 *T)不仅影响性能和内存行为,更关键的是决定了能否修改原始结构体字段以及是否满足接口实现要求。你提供的示例中 SayHi() 仅读取字段(h.name, h.phone),因此无论用 func (h Human) SayHi() 还是 func (h *Human) SayHi(),输出都相同——但这只是“表面一致”,背后语义截然不同。
当使用 func (h *Human) SayHi() 时,h 是指向原始 Human 实例的指针。这意味着:
若改为 func (h Human) SayHi(),每次调用都会创建 Human 的完整副本。例如:
func (h *Human) GrowOld() {
h.age++ // ✅ 成功修改原始 age 字段
}
func (h Human) TryGrowOld() {
h.age++ // ❌ 只修改副本,原始 age 不变
}验证差异的完整示例:
package main
import "fmt"
type Human struct {
name string
age int
}
// 指针接收者:可修改原值
func (h *Human) GrowOld() {
h.age++
}
// 值接收者:仅修改副本
func (h Human) CopyGrowOld() {
h.age++
}
func main() {
mark := Human{"Mark", 25}
fmt.Printf("Before: %v\n", mark) // {Mark 25}
mark.GrowOld() // ✅ age 变为 26
mark.CopyGrowOld() // ❌ 无影响
fmt.Printf("After: %v\n", mark) // {Mark 26}
}
天然线程安全;总结:* 不是语法装饰,而是 Go 类型系统中「可变性契约」的显式声明。选择 *T 还是 T,本质是在回答:“这个方法是否需要改变调用者的状态?”——答案为是,则必用指针接收者。