Go 接口是隐式实现。 对于一个数据类型,无需声明它实现了哪些接口,只需要实现接口必需的方法即可。当然了,存在一个小问题就是: 我们可能无意间实现了某个接口:) ,所以 命名
是多么重要的一件事情。
type 接口名称 interface { 方法1名称(参数列表 ...) 返回值列表... 方法2名称(参数列表 ...) 返回值列表... 方法3名称(参数列表 ...) 返回值列表... ... ... }
package main import ( "fmt" "math" ) // 声明一个图形接口 type geometry interface { area() float64 perimeter() float64 } // 多个字段类型相同时,可以并列声明 type rectangle struct { width, height float64 } type circle struct { radius float64 } // rectangle 隐式实现了 geometry 接口的 area 方法 func (r *rectangle) area() float64 { return r.width * r.height } // rectangle 隐式实现了 geometry 接口的 perimeter 方法 func (r *rectangle) perimeter() float64 { return (r.width + r.height) * 2 } // circle 隐式实现了 geometry 接口的 area 方法 func (c *circle) area() float64 { return math.Pi * c.radius * c.radius } // circle 隐式实现了 geometry 接口的 perimeter 方法 func (c *circle) perimeter() float64 { return 2 * math.Pi * c.radius } func main() { r := &rectangle{ width: 10, height: 5, } fmt.Printf("Rectangle area = %.2f, perimeter = %.2f \n", r.area(), r.perimeter()) c := &circle{ radius: 10, } fmt.Printf("Circle area = %.2f, perimeter = %.2f \n", c.area(), c.perimeter()) } // $ go run main.go // 输出如下 /** Rectangle area = 50.00, perimeter = 30.00 Circle area = 314.16, perimeter = 62.83 */
方法的声明和普通函数的声明类似,只是在函数名字前面多了一个 接收者参数
(接收者参数将方法绑定到其对应的数据类型上)。方法可以绑定到任何数据类型上,但是大多数情况下,绑定的都是 结构体。
func (接收者参数) 方法名称(参数列表 ...) 返回值列表... { // do something }
package main import "fmt" type person struct { name string age int16 } func (p person) sayName() { fmt.Printf("Hi, my name is %s\n", p.name) } func (p person) sayAge() { fmt.Printf("Hi, my age is %d\n", p.age) } func main() { tom := &person{ name: "Tom", age: 6, } tom.sayName() tom.sayAge() } // $ go run main.go // 输出如下 /** Hi, my name is Tom Hi, my age is 6 */
相比结构体方法,指针结构体方法除了将方法参数变为指针外,在引用对应的字段时,无需加 *
标识符, 这一点和普通指针变量引用时有所区别,需要注意。
package main import "fmt" type person struct { name string age int16 } func (p *person) sayName() { // 结构体为指针类型 fmt.Printf("Hi, my name is %s\n", p.name) } func (p *person) sayAge() { // 结构体为指针类型 fmt.Printf("Hi, my age is %d\n", p.age) } func main() { tom := &person{ name: "Tom", age: 6, } tom.sayName() tom.sayAge() } // $ go run main.go // 输出如下 /** Hi, my name is Tom Hi, my age is 6 */
编译器会对方法的 接收者参数
进行检查,具体来说:
• 接收者形参为普通变量类型
• 实参为普通变量类型,编译正常
• 实参为指针变量类型,编译正常
• 接收者形参为指针变量类型
• 实参为普通变量类型,编译报错
• 实参为指针变量类型,编译正常