初学go时很多同学会把 值接收者 和 指针接收者 的方法相互调用搞混淆,好多同学都只记得指针类型可以调用值接收者方法指针接收者方法,而值类型只能调用值接收者方法,其实不然,在某些情况下,值类型也是可以调用指针接收者方法的。

最近又看到有同学发出了这样的疑问,所以打算记录一下,用以备忘、分享。

类型不同可以调用

package main

import (
"fmt"
) type field struct {
name string
} func (p *field) pointerMethod() {
fmt.Println(p.name)
} func (p field) valueMethod() {
fmt.Println(p.name)
} func main() {
fp := &field{name: "pointer"}
fv := field{name: "value"} fp.pointerMethod()
fp.valueMethod()
fv.pointerMethod()
fv.valueMethod()
}
//output: pointer pointer value value

首先我们给field定义了两个方法,一个是指针接收者的pointerMethod,一个是值接收者valueMethod

然后我们创建了变量,fp是指针类型,fv是值类型。

fpfv分别调用pointerMethodvalueMethod,可以看到他们都可以通过编译正常输出。

当类型和方法的接收者类型不同时,编译器会做一些操作:

值类型调用指针接收者方法时,实际为(&fv).pointerMethod()

指针类型调用值接收者方法时,实际为(*fp).valueMethod()

这里是类型不同可以相互调用的情况,再说下不能调用的情况。

类型不同不可以调用

不能调用的情况有两种:

  • 值类型不能被寻址
  • 指针接收者实现接口

两种情况都是值类型不能调用指针接收者方法

值类型不能被寻址

如果值类型实体不能被寻址,那么它就不能调用指针接收者方法

package main

import (
"fmt"
) type field struct {
name string
} func (p *field) pointerMethod() {
fmt.Println(p.name)
} func (p field) valueMethod() {
fmt.Println(p.name)
} func NewFiled() field {
return field{name: "right value struct"}
} func main() {
NewFiled().valueMethod()
NewFiled().pointerMethod()
}

运行代码报错:

./x.go:37:12: cannot call pointer method on NewFiled()
./x.go:37:12: cannot take the address of NewFiled()

看来编译器首先试着给 NewFoo() 返回的右值调用 pointer method,出错;然后试图给其插入取地址符,未果,就只能报错了。

至于左值和右值的区别,大家感兴趣可以自行搜索一下。大致来说,最重要区别就是是否可以被寻址,可以被寻址的是左值,既可以出现在赋值号左边也可以出现在右边;不可以被寻址的即为右值,比如函数返回值、字面值、常量值等等,只能出现在赋值号右边。

指针接收者实现接口

使用指针接收者实现接口方法,那么只有指针类型的实体实现了接口

package main

import "fmt"

type human interface {
speak()
sing()
} type man struct {
} func (m man) speak() {
fmt.Println("speaking")
} func (m *man) sing() {
fmt.Println("singing")
} func main() {
var h human = &man{}
h.speak()
h.sing()
}

上面代码可以正常编译输出,但是我们把&man{}修改为man{}就编译不过了

func main() {
var h human = man{}
h.speak()
h.sing()
}

报错如下:

./x.go:22:6: cannot use man literal (type man) as type human in assignment:
man does not implement human (sing method has pointer receiver)

man没有实现human,因为sing是个指针接收者方法。

这里看下飞雪无情的总结:

实体类型以指针接收者实现接口的时候,只有指向这个类型的指针才被认为实现了该接口

如果是值接收者,实体类型的值和指针都可以实现对应的接口;如果是指针接收者,那么只有类型的指针能够实现对应的接口

再看下饶大的解释:

接收者是指针类型的方法,很可能在方法中会对接收者的属性进行更改操作,从而影响接收者;而对于接收者是值类型的方法,在方法中不会对接收者本身产生影响。

所以,当实现了一个接收者是值类型的方法,就可以自动生成一个接收者是对应指针类型的方法,因为两者都不会影响接收者。但是,当实现了一个接收者是指针类型的方法,如果此时自动生成一个接收者是值类型的方法,原本期望对接收者的改变(通过指针实现),现在无法实现,因为值类型会产生一个拷贝,不会真正影响调用者。

平时我们写代码的时候也不用可以记这个,不仅编译器会报错,goland也一样会提示。

参考

https://www.qtmuniao.com/2020/01/06/go-value-pointer-method/

https://qcrao91.gitbook.io/go/interface/zhi-jie-shou-zhe-he-zhi-zhen-jie-shou-zhe-de-qu-bie

https://www.flysnow.org/2017/04/03/go-in-action-go-interface.html

golang指针接收者和值接收者方法调用笔记的更多相关文章

  1. Golang 方法接收者是值还是指针问题

    对于普通结构体作为接收者,值和指针并没有区别. (以下代码摘抄自Go In Action 中文版) type defaultMatcher struct{} // 方法声明为使用 defaultMat ...

  2. go中方法的接收者是值或者指针的区别

    值类型的变量和指针类型的变量 先声明一个结构体: type T struct { Name string } func (t T) M1() { t.Name = "name1" ...

  3. atitit.架构设计---方法调用结果使用异常还是返回值

    atitit.架构设计---方法调用结果使用异常还是返回值 1. 应该返回BOOL类型还是异常 1 2. 最终会有四种状况,抛出异常.返回特殊值.阻塞.超时 1 3. 异常的优缺点点 1 4. jav ...

  4. Java加载jar文件并调用jar文件当中有参数和返回值的方法

    在工作当中经常遇到反编译后的jar文件,并要传入参数了解其中的某些方法的输出,想到Java里面的反射可以实现加载jar文件并调用其中的方法来达到自己的目的.就写了个Demo代码. 以下的类可以编译生成 ...

  5. JAVA方法调用中的解析与分派

    JAVA方法调用中的解析与分派 本文算是<深入理解JVM>的读书笔记,参考书中的相关代码示例,从字节码指令角度看看解析与分派的区别. 方法调用,其实就是要回答一个问题:JVM在执行一个方法 ...

  6. 深入解析多态和方法调用在JVM中的实现

    深入解析多态和方法调用在JVM中的实现 1. 什么是多态 多态(polymorphism)是面向对象编程的三大特性之一,它建立在继承的基础之上.在<Java核心技术卷>中这样定义: 一个对 ...

  7. golang中的左值VS右值

    对应关系 左值 可寻址 右值 不可寻址 可寻址:可以通过&取地址符,获取内存地址; 可寻址,也就是分配了内存; 不可寻址:根本没有分配内存; 常量const 常量通常只支持数字/字符串/布尔, ...

  8. 图解JVM执行引擎之方法调用

    一.方法调用 方法调用不同于方法执行,方法调用阶段的唯一任务就是确定被调用方法的版本(即调用哪一个方法),暂时还不涉及方法内部的具体运行过程.Class文件的编译过程中不包括传统编译器中的连接步骤,一 ...

  9. [Effective JavaScript 笔记]第18条:理解函数调用、方法调用及构造函数调用之间的不同

    面向对象编程中,函数.方法.类的构造函数是三种不同的概念. JS中,它们只是单个构造对象的三种不同的使用模式. 三种不同的使用模式 函数调用 function hello(username){ ret ...

随机推荐

  1. Linux-shell编程经验记录

    Linux-shell编程经验总结 1.接收用户输入 #读取用户输入并且将输入保存到input变量中 read -p "请输入:" input #也可以先输出信息再进行读取,这里的 ...

  2. GitHub标星8k,字节跳动高工熬夜半月整理的“组件化实战学习手册”,全是精髓!

    前言 什么是组件化? 最初的目的是代码重用,功能相对单一或者独立.在整个系统的代码层次上位于最底层,被其他代码所依赖,所以说组件化是纵向分层. 为什么要使用组件化? 当我们的项目越做越大的时候,有时间 ...

  3. Spring对Controller、Service、Dao进行Junit单元测试总结

    测试类注解 @RunWith(SpringRunner.class) @SpringBootTest(classes={DemoApplication.class}) 以Controller层的的单元 ...

  4. MapReduce框架原理-Writable序列化

    序列化和反序列化 序列化就是把内存中的对象,转换成字节序列(或其他数据传输协议)以便于存储(持久化)和网络传输. 反序列化就是将收到字节序列(或其他数据传输协议)或者是硬盘的持久化数据,转换成内存中的 ...

  5. Golang语言系列-16-context上下文

    context上下文 控制子goroutine退出 全局变量方式 package main import ( "fmt" "sync" "time&q ...

  6. iOS开发之Video转GIF

    前言 最近遇到需要将video转化为gif的问题,网上找的在线转换限制太多,索性就自己写了一个工具APP.文章末尾有开源代码和打包好的APP,如有需要请自行下载. 效果图 核心代码 来源 class ...

  7. 42岁大龄程序员的迷茫,看我最新尝鲜.net 5+Dapper搭建的WebAPI框架

    42岁大龄程序员的迷茫 我真傻,真的.我单知道雪天是野兽在深山里没有食吃,会到村里来;我不知道春天也会有-- 我真傻,真的.我单知道程序员要活到老学到老,年龄大了要失业;我不知道码农(新型农民工)也会 ...

  8. 004 Ethernet(以太网)详解

    一.以太网 以太网是一种计算机局域网技术.IEEE组织的IEEE 802.3标准制定了以太网的技术标准,它规定了包括物理层的连线.电子信号和介质访问层协议的内容. 以太网有两类:第一类是经典以太网,第 ...

  9. S3C2440—10.代码重定位

    文章目录 一.启动方式 1.1 NAND FLASH 启动 1.2 NOR FLASH 启动 二. 段的概念 2.1 重定位数据段 2.2 加载地址的引出 三.链接脚本 3.1 链接脚本的引入 3.2 ...

  10. ☕【Java技术指南】「OpenJDK专题」想不想编译属于你自己的JDK呢?(Windows10环境)

    Win10下编译OpenJDK8 编译环境 Windows10专业版64位: 编译前准备 Tip: 以下软件的安装和解压目录尽量不要包含中文或空格,不然可能会出现问题 安装 Visual Studio ...