脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服务器之家 - 脚本之家 - Golang - Go方法接收者值接收者与指针接收者详解

Go方法接收者值接收者与指针接收者详解

2022-12-02 14:52大愚Talk Golang

这篇文章主要为大家介绍了Go方法接收者值接收者与指针接收者详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

引言

在review 一些代码中,发现经常某个类型定义的方法,其接收者既有值类型,又有指针类型,然后 Goland 就有提示: Struct Person has methods on both value and pointer receivers. Such usage is not recommended by the Go Documentation.

一般来讲,这个提示对代码的运行并不会产生什么问题。只不过对于有轻微 “代码洁癖” 的人来讲,体感不好,就一定想要改统一。

当然,我并不是想讲要统一的问题,前面说这么多废话,只是为了铺垫一下引出本文的内容:Go中的值接收者与指针接收者有什么关系与区别,该怎么选?

联系与区别

在继续讲下去之前,我们得先明确,Go 里边能够定义方法的必须是自定义类型,而不能是系统内置类型,比如 int、string 这种是不可以为其添加方法的。

那么当我们定义了一个自定义类型,可以为其添加方法,先上代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
package main
import "fmt"
type Person struct {
   name string
   age  int
}
// 值针接收者
func (p Person) GetName() string {
   return p.name
}
// 指针接收者
func (p *Person) GetAge() int {
   return p.age
}
func main() {
   //  定义了一个【值类型】
   t := Person{
      name: "DaYu",
      age:  int(28),
   }
   // 调用值方法
   fmt.Println(t.GetName())
   // 调用指针方法
   fmt.Println(t.GetAge())
}
-----运行结果-------
study/demo01/client go run *
DaYu
28

指针类型调用结果

从使用过程看,值类型的变量,可以调用该类型的值接收者方法,也可以调用指针接收者方法。

反之,我们可以定义一个指针类型,然后看看调用结果:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package main
import "fmt"
type Person struct {
   name string
   age  int
}
func (p *Person) GetName() string {
   return p.name
}
func (p Person) GetAge() int {
   return p.age
}
func main() {
   // 注意,其它地方都没有改,只是这里改变了类型
   t := &Person{
      name: "DaYu",
      age:  int(28),
   }
   fmt.Println(t.GetName())
   fmt.Println(t.GetAge())
}
-----运行结果-------
study/demo01/client go run *
DaYu
28

这段代码告诉我们,指针类型的变量,可以调用该类型的值接收者方法,也可以调用指针接收者方法。

是不是特别有意思?

  • 值类型变量,可以调用值接收的方法,也可以调用指针接收者的方法;
  • 指针类型变量,可以调用值接收的方法,也可以调用指针接收者的方法。

看起来好像两者对等的,并没有差别。那么二者真的没有差别吗?只是一种表达形式上的差异?其实不然,如果引入接口类型后,我们再来看看。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
package main
// 新增的接口
type Animal interface {
   GetName() string
   GetAge() int
}
type Person struct {
   name string
   age  int
}
func (p *Person) GetName() string {
   return p.name
}
func (p Person) GetAge() int {
   return p.age
}
func main() {
   // 定义的接口变量
   var ani Animal
   // person 实现了 Animal 接口,赋值给了 ani 变量
   // 但是,这里编译会通不过,错误如下:
   // Cannot use 'Person{ name: "DaYu", age: int(28), }' (type Person) as the type Animal Type does not implement 'Animal' as the 'GetName' method has a pointer receiver
   ani = Person{
      name: "DaYu",
      age:  int(28),
   }
   ani.GetName()
   ani.GetAge()
}

为什么会报错呢? 错误提醒很明显了:Person 没有实现 Animal 的 GetName 方法。因为在上面的代码中,我们实现 GetName 方法的是 (*Person) 类型。

但是为什么 GetAge 方法不报错呢? 那是因为 Go 里边对于 (Type)Method 的方法,会自动让他拥有 (*Type)Method 方法的能力。

实现接口时约束

  • 如果定义的是 (Type)Method,则该类型会隐式的声明一个 (*Type)Method;
  • 如果定义的是 (*Type)Method ,则不会隐式什么一个 (Type)Method。

至于为什么不也隐式申明一个 (Type)Method ,我觉得有一个原因是,我们一般采用指针接收者时,方法内部改变的值,接收者本身也会改变,那么此时如果隐式有这样一个申明,外部使用值类型时,这个改变就不会生效,语义上就会非常奇怪。

该怎么用

从使用表现上看,指针接收者在方法内部的改变,会体现到其本身。但这并不是决定我们要不要用指针接收者的唯一理由! 最重要的还是看接收者要不要全局共享一个实体,其次某些场景下,如果接收者本身太大,拷贝成本很高,也应该使用指针接收者。

回到文档开篇的问题,为什么不建议值接收者、指针接收者混用,主要还是在于语义不够清晰,存在潜在理解成本的问题。

以上就是Go方法接收者值接收者与指针接收者详解的详细内容,更多关于Go方法值接收者指针接收者的资料请关注服务器之家其它相关文章!

原文链接:https://juejin.cn/post/7158354728961703973

延伸 · 阅读

精彩推荐
  • Golang一篇带给你Go语言的并发

    一篇带给你Go语言的并发

    并行指的是在同一时间,多个程序在不同的 CPU 上共同运行,互相之间并没有对 CPU 资源进行竞争。比如,我在看书的时候,左手用来翻书,右手做笔记,两...

    自然醒的笔记本3722021-06-24
  • Golang手把手教你vscode配置golang开发环境的步骤

    手把手教你vscode配置golang开发环境的步骤

    这篇文章主要介绍了手把手教你vscode配置golang开发环境的步骤,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以...

    平凡键客6872021-04-18
  • GolangGo语言错误处理异常捕获+异常抛出

    Go语言错误处理异常捕获+异常抛出

    这篇文章主要介绍了Go语言错误处理异常捕获和异常抛出,Go语言的作者认为java等语言的错误处理底层实现较为复杂,就实现了函数可以返回错误类型以及简...

    酷尔。6872022-02-25
  • GolangGo语言实现操作MySQL的基础知识总结

    Go语言实现操作MySQL的基础知识总结

    这篇文章主要总结一下怎么使用Go语言操作MySql数据库,文中的示例代码讲解详细,需要的朋友可以参考以下内容,希望对大家有所帮助...

    yi个俗人4532022-11-13
  • GolangGO语言中的方法值和方法表达式的使用方法详解

    GO语言中的方法值和方法表达式的使用方法详解

    这篇文章主要介绍了GO的方法值和方法表达式的使用方法,本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下 ...

    周伯通的麦田5272020-06-06
  • GolangGolang通过包长协议处理TCP粘包的问题解决

    Golang通过包长协议处理TCP粘包的问题解决

    本文主要介绍了Golang通过包长协议处理TCP粘包的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋...

    地下十一楼的森琦11442022-10-25
  • GolangGolang监听日志文件并发送到kafka中

    Golang监听日志文件并发送到kafka中

    这篇文章主要介绍了Golang监听日志文件并发送到kafka中,日志收集项目的准备中,本文主要讲的是利用golang的tail库,监听日志文件的变动,将日志信息发送...

    zhijie3472022-09-16
  • Golang浅谈go中defer的一个隐藏功能

    浅谈go中defer的一个隐藏功能

    这篇文章主要介绍了浅谈go中defer的一个隐藏功能,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面...

    KevinYan3492020-06-01