理解Go语言中的方法和接收者

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;
    public 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('dary1', 'dary1@example');
$user->changeEmail('[email protected]');
$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,它是指针接收者方法。可以看到,值接收者方法,接收者是一个副本,无法修改;指针接收者是引用,可以修改。

我们再来看一下调用:

扫描二维码关注公众号,回复: 11499873 查看本文章
dary1 := {"dary1", "[email protected]"}
dary1.changeEmail("[email protected]")
dary1.notify()

看看,是不是很熟悉!对,就像我们刚刚写过的php代码一样,有没有!dary1就是对象,name和email就是属性,notify和changeEmail就是它的方法。只是,不同的是,我们没有将它放到class中,而是用另外一种方式让它们结合了,有了关系!

关于值接收者和指针接收者,其实Go在变异的时候有一个隐式转换,将其转换为正确的接收者类型。就像下面这样。

//daryl.changeEmail("[email protected]")
(&daryl).changeEmail("[email protected]")

wife := &daryl
//wife.notify()
(*wife).notify()

猜你喜欢

转载自blog.csdn.net/zhangge3663/article/details/107181870