原文链接: Go 专栏|说说方法

最近又搬家了,已经记不清这是第几次搬家了。搬到了公司附近,走路十分钟,以后加班可方便了。

这一篇来说一说方法,方法可以看作是某种特定类型的函数,是 Go 面向对象编程的第一步。用好方法,具备面向对象编程思想是关键。

声明

方法的声明和函数类似,他们的区别是:方法在定义的时候,会在 func 和方法名之间增加一个参数,这个参数就是接收者,这样我们定义的这个方法就和接收者绑定在了一起,称之为这个接收者的方法。

type Person struct {
    name string
}

func (p Person) String() string {
    return "person name is " + p.name
}

func 和方法名之间增加的参数 (p Person) 就是接收者。现在我们说,类型 Person 有了一个 String 方法。

调用方法非常简单,使用类型的变量和 . 操作符进行调用即可。

p := Person{name: "zhangsan"}

// 调用方法
fmt.Println(p.String()) // person name is zhangsan

值语义和引用语义

Go 语言里有两种类型的接收者:值接收者和指针接收者。

使用值类型接收者定义的方法,在调用的时候,使用的其实是值接收者的一个副本,所以对该值的任何操作,不会影响原来的类型变量。

func main() {
    p := Person{name: "zhangsan"}

    // 调用方法
    fmt.Println(p.String()) // person name is zhangsan

    // 值接收者
    p.Modify()
    fmt.Println(p.String()) // person name is zhangsan
}

// 值接收者
func (p Person) Modify() {
    p.name = "lisi"
}

接下来再看一下使用指针接收者的效果:

func main() {
    p := Person{name: "zhangsan"}

    // 调用方法
    fmt.Println(p.String()) // person name is zhangsan

    // 指针接收者
    p.ModifyP()
    fmt.Println(p.String()) // person name is lisi
}

// 指针接收者
func (p *Person) ModifyP() {
    p.name = "lisi"
}

可以看到,改变了原始值,其实这一点和函数传参是一样的。

有没有发现,我们在调用指针接收者方法的时候,使用的也是一个值的变量,并不是一个指针,正常来说应该这么写:

(&p).ModifyP()
fmt.Println(p.String())

同样的,如果是一个值接收者的方法,使用指针也是可以调用的:

(&p).Modify()
fmt.Println(p.String())

原因是编译器帮我们自动转义了,这一点大大的方便了我们开发者。

方法变量和表达式

上文中已经介绍了一种调用方法,直接使用 . 操作符,比如:p.String()

接下来再介绍两种调用方法:

方法变量

p.Add 可以赋值给一个方法变量,它相当于一个函数,把方法绑定到一个接收者上。然后函数只需要提供实参而不需要提供接收者即可调用。

type Point struct {
    x, y int
}

func main() {
    // 方法变量
    p1 := Point{1, 2}
    q1 := Point{3, 4}
    f := p1.Add
    fmt.Println(f(q1)) // {4 6}
}

func (p Point) Add(q Point) Point {
    return Point{p.x + q.x, p.y + q.y}
}

方法表达式

方法表达式写成 T.f 或者 (*T).f,其中 T 是类型,是一种函数变量。

因为调用方法必须要提供接收者,这种方法相当于把接收者替换成了函数的第一个形参,因此它可以像函数一样调用。

// 方法表达式
f1 := Point.Add
fmt.Println(f1(p1, q1)) // {4 6}

总结

本文主要学习了 Go 的方法,方法的声明和函数类似,他们的区别是:方法在定义的时候,会在 func 和方法名之间增加一个参数,这个参数就是接收者。

接收者有两种类型:值接收者和指针接收者。不管是使用值接收者,还是指针接收者,一定要搞清楚类型的本质:对类型进行操作的时候,是要改变当前值,还是要创建一个新值进行返回?这些就可以决定我们是采用值传递,还是指针传递。

最后就是方法的调用,可以直接使用 . 操作符调用,还可以使用方法变量和方法表达式。

只有基于面向对象编程思想,才能使用好方法。在后面要学习的接口中,方法还有更多的应用。


文章中的脑图和源码都上传到了 GitHub,有需要的同学可自行下载。

地址: https://github.com/yongxinz/g...

关注公众号 AlwaysBeta,回复「goebook」领取 Go 编程经典书籍。

Go 专栏文章列表:

  1. Go 专栏|开发环境搭建以及开发工具 VS Code 配置
  2. Go 专栏|变量和常量的声明与赋值
  3. Go 专栏|基础数据类型:整数、浮点数、复数、布尔值和字符串
  4. Go 专栏|复合数据类型:数组和切片 slice
  5. Go 专栏|复合数据类型:字典 map 和 结构体 struct
  6. Go 专栏|流程控制,一网打尽
  7. Go 专栏|函数那些事
  8. Go 专栏|错误处理:defer,panic 和 recover
  9. Go 专栏|说说方法
  10. Go 专栏|接口 interface

alwaysbeta
84 声望13 粉丝