Go中为struct添加方法需通过接收者实现:值接收者(u User)复制结构体,指针接收者(u *User)可修改原字段;指针接收者方法必须用可寻址变量调用,且影响接口实现与嵌入方法提升。
Go 语言不支持传统面向对象的“类方法”,而是通过为类型(包括 struct)定义**接收者(receiver)**来实现类似行为。关键点是:方法必须定义在同一个包内,且接收者类型不能是内置类型(如 int、string)或别名之外的非本地类型。
基本语法如下:
func (r ReceiverType) MethodName(params) (results) {
// 方法体
}
其中 ReceiverType 必须是该 struct 的**具体类型**(如 User)或其指针类型(如 *User),不能是接口或未定义类型。
(u User) 表示值接收者:调用时会复制整个 struct,适合小对象且不需修改原值(u *User) 表示指针接收者:可修改原始 struct 字段,也避免大 struct 复制开销当方法需要修改 struct 的字段时,必须使用指针接收者。值接收者操作的是副本,对原 struct 无影响。
常见错误现象:cannot assign to struct field ... in function call 或字段看似没变——本质是改了副本。
u.Name = "Alice")只在 *User 接收者下生效u.Method() 等价于 (&u).Method()),但前提是 u 是可寻址变量(不能是字面量或临时值)Go 中接口是否被满足,取决于类型的方法集(method set)。这点极易被忽略:
type T struct{} 的方法集只包含值接收者方法*T 的方法集包含值接收者 + 指针接收者方法*T 能实现它,T 不能例如,标准库 io.Writer 的 Write([]byte) (int, error) 是指针接收者风格,所以 os.File 类型实际是以指针形式参与接口赋值。
典型陷阱: v 编译失败,但 
var w io.Writer = &MyStruct{} 成功——差一个 & 就无法满足接口。
Go 不支持继承,但可通过匿名字段“组合”行为。嵌入 struct 后,其方法会“提升”到外层 struct 的方法集中,但有严格限制:
Logger,它有 func (l *Logger) Log(),那么只有 *Outer 才能调用 outer.Log();Outer 值类型无法调用o.Logger.Log()
这说明:方法提升不是无条件的“继承”,而是基于接收者类型的精确匹配。很多人卡在这里,以为嵌入就等于自动获得全部能力。
真正复杂的地方在于:接收者类型、方法集、接口实现、嵌入提升四者交织,稍不注意就会出现“明明写了方法却不能调用”或“接口赋值失败”的问题。调试时优先检查接收者是指针还是值,再确认变量是否可寻址、接口定义是否匹配方法签名。