

新闻资讯
技术教程必须用指针接收者:修改字段、大结构体、实现接口、统一风格——语义上表示“方法会改变接收者”,避免副本无效修改与接口实现失败。
如果方法内部要改结构体的任何字段,*T 是唯一选择。值接收者 T 操作的是副本,改了也白改。
func (u User) SetName(n string) { u.Name = n } → 调用后 u.Name 不变func (u *User) SetName(n string) { u.Name = n } → 真正更新原对象int 字段,只要语义上是“更新状态”,就该用指针Go 会完整拷贝值接收者的整个结构体。字段多、含切片/映射/大数组时,复制开销明显。
type Point { X, Y int })值接收者可接受[]byte、map[string]int)建议统一用 *T
接口实现与否,取决于类型的方法集是否包含所需方法。而方法集由接收者类型决定:
T 的方法集只含值接收者方法*T 的方法集含值接收者 + 指针接收者方法func (t *T) Read() []byte,则只有 *T 类型能赋给 io.Reader 接口var t T; var r io.Reader = t 编译失败,但 var r io.Reader = &t 可行Go 允许 t.Method() 调用指针接收者方法(自动转成 &t.Method()),前提是 t 是可寻址的变量。
Person{}.SetName("x") 或 make([]Person, 1)[0].SetName("x")
*T,其余全用 *T —— 避免调用方猜“这个能不能改”type Config struct {
Timeout int
Hosts []string
}
// ✅ 统一指针接收者,清晰、安全、可扩展
func (c *Config) SetTimeout(t int) { c.Timeout = t }
func (c *Config) AddHost(h string) { c.Hosts = append(c.Hosts, h) }
func (c *Config) String() string { return fmt.Sprintf("timeout=%d, hosts=%v", c.Timeout, c.Hosts) }
指针接收者不是性能优化技巧,而是