结构体通过实现接口中定义的方法来实现接口。
- 结构体赋值接口
package main
import "fmt"
type Phone interface {
call(name string) string
}
type Person struct {
Name string
}
//绑定方法
func (p *Person) Test2(newName string) string {
p.Name = newName
return p.Name
}
//实现接口
func (p Person) call(name string) string {
return p.Name + "打电话给" + name
}
func main() {
// 赋值给结构体变量
// var p1 Person
p1 := Person{"abc"}
str1 := p1.call("xyz")
fmt.Println(str1) //abc打电话给xyz
// 赋值给接口变量
var p2 Phone
p2 = Person{"abc"}
str2 := p2.call("xyz")
fmt.Println(str2) //abc打电话给xyz
}
// 如果一个struct实现了interface中的所有方法,那么可以将一个struct的变量赋值给一个interface,只实现一部分方法,那么就不能赋值操作。
// 这个例子中,如果Phone中还有一个demo()方法,但是person没有实现这个demo(),那么在主函数中执行赋值就会出错,提示没有实现notifier的所有接口。
2. 接口组合
package main
import "fmt"
type Person struct {
Name string
}
type notifier interface {
notice()
}
type informer interface {
inform()
}
type tellA interface {
notifier
informer
}
type tellB interface {
notice()
inform()
}
func (p Person) notice() {
fmt.Println(p.Name, "noticing")
}
func (p Person) inform() {
fmt.Println(p.Name, "informing")
}
func main() {
var p tellA //注意p是组合接口的类型
p = Person{"abc"}
p.notice() //abc noticing
var p1 tellB //注意p1是组合接口的类型
p1 = Person{"abc"}
p1.notice() //abc noticing
}
// 只要两个接口的拥有相同的方法列表,方法的次序可以不同,那么这两个接口是等价的,可以相互赋值。
// 所以说上面tell组合接口的两种形式是一样的。
3. 多态
package main
import "fmt"
type Dog struct {
}
type Cat struct {
}
type Animal interface {
sound()
}
func (d Dog) sound() { // 这里必须是结构体receiver,不能是指针receiver
fmt.Println("dog: wang wang")
}
func (c Cat) sound() {
fmt.Println("cat: miao miao")
}
//接收一个实现了Animal接口的struct
func Jiao(a Animal) {
a.sound()
}
func main() {
a1 := Dog{}
a2 := Cat{}
Jiao(a1) //dog: wang wang
Jiao(a2) //cat: miao miao
}
// 其实多态最主要的就是接受的那个参数类型是一个接口类型,并且是多个结构体实现了的接口,结合前面一个示例,那么实现了接口的结构体实例可以赋值给接口类型的变量,从而调用接口中定义的方法。
// 这里会涉及一个空接口,因为空接口内部什么都没有,一个方法都没有,那么皆可以认为所有的结构体都实现了这个空接口。那么所有结构体都可以赋值给这个空接口的实例。