首页
学习
活动
专区
工具
TVP
发布

Go 系列教程——第17 部分:方法

欢迎来到 Golang 系列教程的第 17 个教程。

什么是方法?

方法其实就是一个函数,在 这个关键字和方法名中间加入了一个特殊的接收器类型。接收器可以是结构体类型或者是非结构体类型。接收器是可以在方法的内部访问的。

下面就是创建一个方法的语法。

(parameter list){}

上面的代码片段创建了一个接收器类型为 的方法 。

方法示例

让我们来编写一个简单的小程序,它会在结构体类型上创建一个方法并调用它。

(){

在线运行程序

在上面程序的第 16 行,我们在 结构体类型上创建了一个 方法。displaySalary()方法在方法的内部访问了接收器 。在第 17 行,我们使用接收器 ,并打印 employee 的 name、currency 和 salary 这 3 个字段。

在第 26 行,我们调用了方法 。

程序输出:。

为什么我们已经有函数了还需要方法呢?

上面的程序已经被重写为只使用函数,没有方法。

在线运行程序

在上面的程序中, 方法被转化为一个函数, 结构体被当做参数传递给它。这个程序也产生完全相同的输出:。

既然我们可以使用函数写出相同的程序,那么为什么我们需要方法?这有着几个原因,让我们一个个的看看。

Go 不是纯粹的面向对象编程语言,而且Go不支持类。因此,基于类型的方法是一种实现和类相似行为的途径。

相同的名字的方法可以定义在不同的类型上,而相同名字的函数是不被允许的。假设我们有一个 和 结构体。可以在 和 上分别定义一个 方法。见下面的程序。

在线运行程序

该程序输出:

上面方法的属性被使用在接口中。我们将在接下来的教程中讨论这个问题。

指针接收器与值接收器

到目前为止,我们只看到了使用值接收器的方法。还可以创建使用指针接收器的方法。值接收器和指针接收器之间的区别在于,在指针接收器的方法内部的改变对于调用者是可见的,然而值接收器的情况不是这样的。让我们用下面的程序来帮助理解这一点。

在线运行程序

在上面的程序中, 方法有一个值接收器 ,而 方法有一个指针接收器 。在 方法中对 结构体的字段 所做的改变对调用者是不可见的,因此程序在调用 这个方法的前后打印出相同的名字。由于 方法是使用指针 接收器的,所以在调用 方法对 字段做出的改变对调用者将是可见的。该程序输出如下:

在上面程序的第 36 行,我们使用 来调用 方法。由于 方法有一个指针接收器,所以我们使用 来调用这个方法。其实没有这个必要,Go语言让我们可以直接使用 。 会自动被Go语言解释为 。

下面的程序重写了,使用 来代替 ,它输出相同的结果。

在线运行程序

那么什么时候使用指针接收器,什么时候使用值接收器?

一般来说,指针接收器可以使用在:对方法内部的接收器所做的改变应该对调用者可见时。

指针接收器也可以被使用在如下场景:当拷贝一个结构体的代价过于昂贵时。考虑下一个结构体有很多的字段。在方法内使用这个结构体做为值接收器需要拷贝整个结构体,这是很昂贵的。在这种情况下使用指针接收器,结构体不会被拷贝,只会传递一个指针到方法内部使用。

在其他的所有情况,值接收器都可以被使用。

匿名字段的方法

属于结构体的匿名字段的方法可以被直接调用,就好像这些方法是属于定义了匿名字段的结构体一样。

在线运行程序

在上面程序的第 32 行,我们通过使用 来访问 结构体的 方法。明确的调用 是没有必要的。该程序输出:

在方法中使用值接收器 与 在函数中使用值参数

这个话题很多Go语言新手都弄不明白。我会尽量讲清楚。

当一个函数有一个值参数,它只能接受一个值参数。

当一个方法有一个值接收器,它可以接受值接收器和指针接收器。

让我们通过一个例子来理解这一点。

在线运行程序

第 12 行的函数 接受一个值参数,方法 接受一个值接收器。

在第 25 行,我们通过值参数 来调用 area 这个函数,这是合法的。同样,我们使用值接收器来调用 area 方法 ,这也是合法的。

在第 28 行,我们创建了一个指向 的指针 。如果我们试图把这个指针传递到只能接受一个值参数的函数 area,编译器将会报错。所以我把代码的第 33 行注释了。如果你把这行的代码注释去掉,编译器将会抛出错误 。这将会按预期抛出错误。

现在到了棘手的部分了,在第35行的代码 使用指针接收器 调用了只接受一个值接收器的方法 。这是完全有效的。原因是当 有一个值接收器时,为了方便Go语言把 解释为 。

该程序将会输出:

在方法中使用指针接收器 与 在函数中使用指针参数

和值参数相类似,函数使用指针参数只接受指针,而使用指针接收器的方法可以使用值接收器和指针接收器。

在线运行程序

在上面程序的第 12 行,定义了一个接受指针参数的函数 。第 17 行定义了一个有一个指针接收器的方法。

在第 27 行,我们调用 perimeter 函数时传入了一个指针参数。在第 28 行,我们通过指针接收器调用了 perimeter 方法。所有一切看起来都这么完美。

在被注释掉的第 33 行,我们尝试通过传入值参数 调用函数 。这是不被允许的,因为函数的指针参数不接受值参数。如果你把这行的代码注释去掉并把程序运行起来,编译器将会抛出错误 。

在第 35 行,我们通过值接收器 来调用有指针接收器的方法 。这是被允许的,为了方便Go语言把代码 解释为 。该程序输出:

在非结构体上的方法

到目前为止,我们只在结构体类型上定义方法。也可以在非结构体类型上定义方法,但是有一个问题。为了在一个类型上定义一个方法,方法的接收器类型定义和方法的定义应该在同一个包中。到目前为止,我们定义的所有结构体和结构体上的方法都是在同一个 包中,因此它们是可以运行的。

在线运行程序

在上面程序的第 3 行,我们尝试把一个 方法添加到内置的类型 。这是不允许的,因为 方法的定义和 类型的定义不在同一个包中。该程序会抛出编译错误 。

让该程序工作的方法是为内置类型 int 创建一个类型别名,然后创建一个以该类型别名为接收器的方法。

在线运行程序

在上面程序的第5行,我们为 创建了一个类型别名 。在第7行,我们定义了一个以 为接收器的的方法 。

该程序将会打印出 。

我已经创建了一个程序,包含了我们迄今为止所讨论的所有概念,详见github。

这就是Go中的方法。祝你有美好的一天。

上一教程 - 结构体

下一教程 - 接口 - I

github 上有本教程的源代码。

  • 发表于:
  • 原文链接https://kuaibao.qq.com/s/20180529G0C6VZ00?refer=cp_1026
  • 腾讯「腾讯云开发者社区」是腾讯内容开放平台帐号(企鹅号)传播渠道之一,根据《腾讯内容开放平台服务协议》转载发布内容。
  • 如有侵权,请联系 cloudcommunity@tencent.com 删除。

扫码

添加站长 进交流群

领取专属 10元无门槛券

私享最新 技术干货

扫码加入开发者社群
领券