接口匹配取决于方法集:值接收者方法可被值和指针调用,指针接收者方法只能由指针调用,因此只有指针拥有完整方法集,赋值给接口时需注意接收者类型。
在Golang中,接口和指针接收者的关系是理解方法集和类型行为的关键。核心在于:Go中的接口调用取决于类型的**方法集**,而方法集又由接收者的类型(值或指针)决定。
Go语言中,一个类型要实现某个接口,必须实现该接口定义的所有方法。但是否能通过值或指针调用这些方法,取决于接收者类型:
这意味着:只有指针拥有全部方法集。值类型的方法集是其本身;而指针类型的方法集包括指向类型的值方法和指针方法。
d.Name
}
type Cat struct {
Name string
}
// 指针接收者
func (c *Cat) Speak() string {
return "Meow! I'm " + c.Name
}
func main() {
// Dog 使用值接收者,值和指针都能赋给接口
var s1 Speaker = Dog{Name: "Buddy"}
var s2 Speaker = &Dog{Name: "Max"}
fmt.Println(s1.Speak()) // Woof! I'm Buddy
fmt.Println(s2.Speak()) // Woof! I'm Max
// Cat 使用指针接收者,只有指针能赋给接口
// var s3 Speaker = Cat{Name: "Luna"} // 错误!Cat 值没有实现 Speak 方法
var s4 Speaker = &Cat{Name: "Luna"}
fmt.Println(s4.Speak()) // Meow! I'm Luna
}
注意:虽然Dog是值接收者,但&Dog{}也能赋给Speaker,因为Go会自动解引用。反过来不行——如果方法是指针接收者,就不能把值赋给接口。
主要出于效率和一致性考虑:
getCat().Speak()中的getCat()返回值)基本上就这些。记住一点:当你定义的方法使用指针接收者时,要把该类型的指针赋给接口;而值接收者更宽松,值和指针都可以。这在构造函数返回指针时特别重要。