Go 快速入门指南 - 接口和方法

接口

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
*/

编译检测

编译器会对方法的 接收者参数 进行检查,具体来说:

• 接收者形参为普通变量类型

 • 实参为普通变量类型,编译正常
  
 • 实参为指针变量类型,编译正常

• 接收者形参为指针变量类型

 • **实参为普通变量类型,编译报错**

 • 实参为指针变量类型,编译正常

联系我

公众号

posted @ 2022-12-22 16:35  洋芋土豆  阅读(79)  评论(0编辑  收藏  举报