当前位置: 首页 > news >正文

go 指针接收者和值接收者的区别

go 指针接收者和值接收者的区别

指针接收者和值接收者的区别主要有两点:

  1. Go 中函数传参是传值,因此指针接收者传递的是接收者的指针拷贝,值接收者传递的是接收者的拷贝---在方法中指针接收者的变量会被修改,而值接收者的成员变量修改是无效的(毕竟传入的是拷贝,修改的自然也是拷贝的成员变量)。
  2. 在接口实现上,如果值接收者实现了某个方法,相当于值接收者和指针接收者都实现了这个方法。反之则不行,即指针接收者实现了某个方法,不能看成值接收者实现这个方法。

注意第2点只影响接口实现,如果不涉及接口的话是不受到这条规则约束的。

一个助记的但是不太正确的观点:接收者是指针类型的方法,很可能在方法中会对接收者的属性进行更改操作,从而影响接收者;而对于接收者是值类型的方法,在方法中不会对接收者本身产生影响。

理解角度:指针接收者可以选择修改自己的值和不修改,值接收者修改自己的值是无效的(因为是拷贝)。所以可以看成指针接收者的能力>值接收者的能力,自然:指针接收者实现某个方法可以看成值接收者实现某个方法,反之则不行。

package mainimport "fmt"type coder interface {code()debug()
}type Gopher struct {language string
}func (p Gopher) code() {fmt.Printf("I am coding %s language\n", p.language)
}func (p *Gopher) debug() {fmt.Printf("I am debuging %s language\n", p.language)
}func main() {var c coder = &Gopher{"Go"} //运行正常//上一行换成: var c coder = Gopher{"Go"} // 则报错 Gopher does not implement coder (debug method has pointer receiver)c.code()c.debug()
}

注意📢:对于👆🏻上面的第 2 点补充:虽然隐式实现的方法不一样,但是如果不涉及接口,单纯在调用的时候,无论是指针接收者还是值接收者实现了某个方法,指针接收者和值接收者都可以调用。本质上是 go 的语法糖。

最佳实践:

如果类型具备“原始的本质”,即其成员都是由 Go 语言里内置的原始类型,如字符串,整型值等,那就定义值接收者类型的方法。

内置的引用类型,如 slice,map,interface,channel,这些类型比较特殊,声明他们的时候,实际上是创建了一个 header​, 对于他们也是直接定义值接收者类型的方法。这样,调用函数时,是直接 copy 了这些类型的 header​,而 header​ 本身就是为复制设计的。

如果类型具备非原始的本质,不能被安全地复制,这种类型总是应该被共享,那就定义指针接收者的方法。

虽然上面是这么说,但是个人感觉在实际使用中,基本可以无脑使用指针接收者。主要原因在于:1.指针接收者没有值拷贝带来的巨大开销。2.如果就想在函数中修改值就必须使用指针接收者。3.值接收者唯一的优势就是为了防止意外的修改,为了防止这一点可以通过创建一个函数或者方法来手动 copy,而这并不会带来多大的开销。

易错点辨析

  • 是否改变结构体的值看的是方法是指针接收者还是值接收者,而不是看调用方是指针还是值。原因在于golang编译器在背后会完成一些工作,比如:解引用,隐式使用引用。代码范例如下:
package mainimport "fmt"type Node struct {val int
}func (receiver Node) changeVal1() {receiver.val++
}func (receiver *Node) changeVal2() {receiver.val++
}
func main() {someOne := Node{}someOne.changeVal1()fmt.Printf("%d\n", someOne.val) //0 ,说明没有改值,因为方法接收者是值接收者someOnePtr := &Node{}someOnePtr.changeVal1()fmt.Printf("%d\n", someOnePtr.val) //0 ,说明没有改值,因为方法接收者是值接收者,与调用方是否指针无关someTwo := Node{}someTwo.changeVal2()fmt.Printf("%d\n", someTwo.val) //1 ,说明成功改值,因为方法接收者是指针接收者,与调用方是否指针无关someTwoPtr := &Node{}someTwoPtr.changeVal2()fmt.Printf("%d\n", someTwoPtr.val) //1 ,说明成功改值,因为方法接收者是指针接收者,与调用方是否指针无关}

参考:https://golang.design/go-questions/interface/receiver/

相关文章:

  • 【Pandas】pandas DataFrame pop
  • 【C++11】列表初始化、右值引用、完美转发、lambda表达式
  • 【adb】bat批处理+adb 自动亮屏,自动解锁屏幕,启动王者荣耀
  • 【C++11】新的类功能、模板的可变参数、包装器
  • ceph集群的pg卡住不同步处理记录【如objects degraded (0.223%)】、pg为unclean和degraded等状态处理
  • 网络原理 - 初识网络 2
  • ubuntu 24.02部署java web服务
  • 基于 Java 的淘宝 API 调用实践:商品详情页 JSON 数据结构解析与重构
  • SerialTest 在Linux环境上安装与编译文档
  • GIS开发笔记(7)结合osg及osgEarth实现不同高度下的三个圆形区域形成的三维覆盖轮廓区域绘制
  • AI 语音公司 ElevenLabs 进军亚太市场设立东京子公司;EverTutor Live :语音交互 AI 教育平台丨日报
  • 应用系统中的报表开发成本知多少?
  • SCA(软件成分分析)技术理念、市场前景及趋势
  • MySQL数据库权限管理
  • 刚刚,DeepSeek公布了推理引擎开源路径,OpenAI也将开始连续一周发布
  • 使用 vxe-table 来格式化任意的金额格式,支持导出与复制单元格格式到 excel
  • AI分析师
  • 高级语言调用C接口(二)回调函数(5)arkts
  • 一站式视频转换,在线转MKV,支持多种格式转换,便捷高效的格式转换之选
  • 客户验收标准模糊,如何明确
  • 久久做bilibili官网网站/亚马逊关键词排名提升
  • 腾宁网络做网站/stp营销战略
  • 网站建设后端/武汉网站排名提升
  • 网站代运营服务/接广告推广
  • 模板网站什么意思/抖音代运营收费详细价格
  • 低代码开发会废了程序员吗/sem优化托管