Go方法接收者值接收者与指针接收者详解
作者:大愚Talk 发布时间:2024-02-17 04:30:45
引言
在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 这种是不可以为其添加方法的。
那么当我们定义了一个自定义类型,可以为其添加方法,先上代码:
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
指针类型调用结果
从使用过程看,值类型的变量,可以调用该类型的值接收者方法,也可以调用指针接收者方法。
反之,我们可以定义一个指针类型,然后看看调用结果:
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
这段代码告诉我们,指针类型的变量,可以调用该类型的值接收者方法,也可以调用指针接收者方法。
是不是特别有意思?
值类型变量,可以调用值接收的方法,也可以调用指针接收者的方法;
指针类型变量,可以调用值接收的方法,也可以调用指针接收者的方法。
看起来好像两者对等的,并没有差别。那么二者真的没有差别吗?只是一种表达形式上的差异?其实不然,如果引入接口类型后,我们再来看看。
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 ,我觉得有一个原因是,我们一般采用指针接收者时,方法内部改变的值,接收者本身也会改变,那么此时如果隐式有这样一个申明,外部使用值类型时,这个改变就不会生效,语义上就会非常奇怪。
该怎么用
从使用表现上看,指针接收者在方法内部的改变,会体现到其本身。但这并不是决定我们要不要用指针接收者的唯一理由! 最重要的还是看接收者要不要全局共享一个实体,其次某些场景下,如果接收者本身太大,拷贝成本很高,也应该使用指针接收者。
回到文档开篇的问题,为什么不建议值接收者、指针接收者混用,主要还是在于语义不够清晰,存在潜在理解成本的问题。
来源:https://juejin.cn/post/7158354728961703973


猜你喜欢
- 问题描述:从网上下了5.7 的MySQL,在bin目录下执行 start mysqld ,弹出个cmd窗口一闪就没了,也看不清是什么报错。m
- 我们先来看下秒杀活动页面代码<!DOCTYPE HTML><html> <head> <
- 使用Python如何操作Redis呢?下面用实例来说明用Python读写Redis数据库。比如,我们插入一条数据,如下:import red
- 本文实例讲述了python将ip地址转换成整数的方法。分享给大家供大家参考。具体分析如下:有时候我们用数据库存储ip地址时可以将ip地址转换
- <script language="javascript" src="js/sett
- 前一篇研究了opencv二值化方法threshold的使用,但是这个方法也存在一定的局限性,假如有一张图存在明显的明暗不同的区域,如下图可以
- 额……首先呢说说这个标题吧,实在不知道叫什么好,因为这个demo呢其实一个艾文王今天中午给丢给我一个图。他说这个是一个面试题,给我看看。这样
- 1. 加法运算示例代码:import torch# 这两个Tensor加减乘除会对b自动进行Broadcastinga = torch.ra
- 过往经验总结注:笔者写本文的目的不是完整细致地描述连接的全过程,而是记录当中遇到的现象、问题,及为什么会产生这个问题的分析。所以部分过程会省
- 使用 datetime 模块中的 timedelta() 方法将天数添加到日期中,例如 result_1 = date_1 + timede
- 不止一次在微信、知乎有读者朋友跑过来问:看完了基础书,甚至看两遍了,但自己写的时候还是没思路,我该怎么办?编程在我看来就是一门手艺活,绝不是
- 首先关键一句话:$(".js-example-tags").select2({ tags:
- 一般在线增加从库有两种方式,一种是通过mysqldump备份主库,恢复到从库,mysqldump是逻辑备份,数据量大时,备份速度会很慢,锁表
- 一个单独的组件注释写了一个组件 加了一些注释效果图如下分页一类的功能都已经写好了 下面就上代码,不知道有几个老哥能看的懂,有不足之处,还望老
- print(X.shape):查看矩阵的行列号print(len(X)):查看矩阵的行数print(X.ndim):查看矩阵的维数1 查看矩
- 借助zookeeper可以实现服务器的注册与发现,有需求的时候调用zookeeper来发现可用的服务器,将任务均匀分配到各个服务器上去.这样
- 一、使用selenium前?1.安装seleniumpip install Selenium2.安装浏览器驱动Chrome驱动文件下载:点击
- 作者: Terrance译者:Sheneyan(子乌)时间:2010.2.6英文原文:13 Useful WordPress SQL Que
- openlayer是目前我们gis常用的一款开源的,并且反馈都特别好的软件了,像之前的ol3, 风靡一时,地图实现也很简单,很实用,目前vu
- 在CSS中我们会经常要用到“清除浮动”Clear,比较典型的就是clear:both;CSS手册上是这样说明的:该属性的值指出了不允许有浮动