0x01 前言
Go 语言的语法实在有些不一样,与其它面向对象语言相比,Go 的方法似乎有些晦涩。
0x02 方法的定义
在 Go 语言里,方法和函数只差了一个,那就是方法在 func
和标识符之间多了一个参数。
type user struct { name string, email string,}//这是函数的定义func notify(email string) { fmt.Println("Email is %s", email)}//这是方法的定义func (u user) notify(email string) { fmt.Println("Email is %d", email)}
我们可以看到,方法是在 func
和 notify
之间多了一个 user
类型的参数 u
,这个 u
就称作接收者。
0x03 接收者
接收者有两种,一种是值接收者,一种是指针接收者。顾名思义,值接收者,是接收者的类型是一个值,是一个副本,方法内部无法对其真正的接收者做更改;指针接收者,接收者的类型是一个指针,是接收者的引用,对这个引用的修改之间影响真正的接收者。像上面一样定义方法,将 user
改成 *user
就是指针接收者。
接收者与对象
相信有很多人看到这个接收者之后都很苦恼,到底这个接收者是什么,是干什么用的。我们在学习一门新的语言的时候,都讲究触类旁通,和我们已经了解的语言作对比。那么我们就通过拿 Go 和其它带有类的面向对象的语言做对比来搞清楚接收者是什么。这里我们用 php
来举例子。
在 php
中,我们要定义一个方法,首先是要定义一个类。
class User{ protected $email; protected $name; poublic function __construct($name, $email) { $this->email = $email; $this->name = $name; } public function notify() { echo "Email is {$email}.\n"; } public function changeEmail($email) { $this->email = $email; }}
然后再实例化一个对象,进行操作,像这样。
$user = new User('daryl', 'daryl@example');$user->changeEmail('daryl@example.com');$user->notify();
接下来,我们参照着来写一下 Go 的方法定义。
首先,我们是先要定义一个类型,比如就是 user
好了,然后我们再定义方法。
type user struct { name string email string}func (u user) notify() { fmt.Println("Email is %d", u.email)}func (u *user) changeEmail(email string) { u.email = email}
我们定义了两个方法,一个是 notify
,它是值接收者方法;还有一个是 changeEmail
,它是指针接收者方法。可以看到,值接收者方法,接收者是一个副本,无法修改;指针接收者是引用,可以修改。
我们再来看一下调用。
daryl := {"daryl", "daryl@oldexample.com"}daryl.changeEmail("daryl@example.com")daryl.notify()
看看,是不是很熟悉!对,就像我们刚刚写过的 php
代码一样,有没有!daryl
就是对象,name
和 email
就是属性,notify
和 changeEmail
就是它的方法。只是,不同的是,我们没有将它放到 class
中,而是用另外一种方式让它们结合了,有了关系!
关于值接收者和指针接收者,其实 Go 在编译的时候有一个隐式转换,将其转换为正确的接收者类型。就像下面这样。
//daryl.changeEmail("daryl@example.com")(&daryl).changeEmail("daryl@example.com")wife := &daryl//wife.notify()(*wife).notify()
0x04 后记
最近在学习 Go 语言,看到有很多人评价 Go 的语法很丑陋,这一点确实不可否认。但是,它的语法有很简单,对于熟悉 C 的人、熟悉含有类的面向对象的语言的人,稍加对比,就能发现其很多相似之处。
上面的都是我自己的拙见,如有错误或者不对的地方,非常欢迎指出!人生总是要不断地去学习嘛~