golang值接收者和指针接收者的区别介绍

作者:蔡蔡217 时间:2024-04-25 13:20:33 

方法

方法能给用户自定义的类型添加新的行为。它和函数的区别在于方法有一个接收者,给一个函数添加一个接收者,那么它就变成了方法。接收者可以是值接收者,也可以是指针接收者
在调用方法的时候,值类型既可以调用值接收者的方法,也可以调用指针接收者的方法;指针类型既可以调用指针接收者的方法,也可以调用值接收者的方法。

package main
import "fmt"
type Person struct {
age int
}
func (p Person) AddAge() {
p.age += 1
}

func (p *Person) GetAge() {
p.age += 1
}

func main() {
// p1 是值类型
p := Person{age: 18}
// 值类型 调用接收者也是值类型的方法
p.AddAge()
fmt.Println(p.age)

// ----------------------

// p2 是指针类型  指针类型调用接收者是值类型的方法
p2 := &Person{age: 100}
p2.AddAge()
fmt.Println(p2.age)
//值类型 调用接收者也是指针类型的方法
p3 := Person{age: 18}
p3.GetAge()
fmt.Println(p3.age)
// 指针类型 调用收者也是指针类型的方法
p4 := Person{age: 100}
p4.GetAge()
fmt.Println(p4.age)
}
//18
//100
//19
//101
值接收者指针接收者
值类型调用者传递一个副本使用值的引用来调用方法
指针类型调用者传递一个副本方法里的操作会影响到调用者,类似于指针传参,拷贝了一份指针

总结:
1.一个结构体的方法的接收者可能是类型值或指针
2.当接受者不是一个指针时,该方法操作对应接受者的值的副本,即使你使用了指针调用函数,但是函数的接受者是值类型,所以函数内部操作还是对副本的操作,而不是指针操作。
3.如果接收者是指针,则调用者修改的是指针指向的值本身。

接口实现

当结构体实现一个接口时,可以在方法中设置一个接收者,比如对于以下接口:

type Inter interface {
   foo()
}

结构体在实现它时,方法的接收者类型可以是:值、指针。比如:

type S struct {}

func (s S) foo() {} // 值类型
func (s *S) foo() {} // 或者指针类型

在使用结构体初始化接口变量时,结构体的类型也可以是:值、指针。比如:

//赋值
var s Inter = S{} // 值类型
s.foo()

var s Inter = &S{} // 指针类型
s.foo()

那么调用接口方法的组合实际有四种情况:
值类型结构体 -> 赋值给接口 -> 调用接收者类型为值类型的结构体方法
指针类型结构体 -> 赋值给接口 -> 调用接收者类型为指针类型的结构体方法
值类型结构体 -> 赋值给接口 -> 调用接收者类型为指针类型的结构体方法(不通过)
指针类型结构体 -> 赋值给接口 -> 调用接收者类型为值类型的结构体方法

结构体类型为值类型、调用了接收者为指针的方法不通过。但是反过来,结构体为指针类型时,却可以调用接收值为值或指针的任何方法。这是为什么呢?

接收者是方法的一个额外的参数,而 Go 在调用函数的时候,参数都是值传递的。将一个指针拷贝,它们还是指向同一个地址,指向一个确定的结构体;将一个值拷贝,它们变成了两个不同的结构体,有着不同的地址。这会导致以下两种情况:

当在一个结构体指针上,通过接口,调用一个接收者为值类型的方法时,Go 首先会创建这个指针的副本,然后将这个指针解引用,再作为接收者参数传递给该方法。这两个指针指向相同的地址,所以它们传递给方法的接收者参数都是相同的。

type Inter interface {
   foo()
}
type S struct {}
func (s S) foo() {} // 接收者为值类型的方法

var a Inter = &S{} // 使用结构体指针初始化一个接口
a.foo() // 调用 foo 方法

// 实际上底层是这样的:
// 首先拷贝 a 的底层值,即 `&S{}`,是一个结构体指针:
var b *S = a.inner_value // a、b 是不同的变量,但是指向同一个结构体
// 然后将 b 解引用,传递给 foo:
foo(*b) // *b 和 *(a.inner_value) 其实都表示同一个结构体

这些规则用来说明是否我们一个类型的值或者指针实现了该接口:

  • 类型 *T 的可调用方法集包含接受者为 *T 或 T 的所有方法集

  • 类型 T 的可调用方法集包含接受者为 T 的所有方法

两者分别在何时使用

如果方法的接收者是值类型,无论调用者是对象还是对象指针,修改的都是对象的副本,不影响调用者;如果方法的接收者是指针类型,则调用者修改的是指针指向的对象本身

使用指针作为方法的接收者的理由:

  • 方法能够修改接收者指向的值。

  • 避免在每次调用方法时复制该值,在值的类型为大型结构体时,这样做会更加高效。

  • 是使用值接收者还是指针接收者,不是由该方法是否修改了调用者(也就是接收者)来决定,而是应该基于该类型的本质。

来源:https://blog.csdn.net/weixin_42128977/article/details/126511071

标签:golang,值接收者,指针接收者
0
投稿

猜你喜欢

  • 一些 T-SQL 技巧

    2024-01-26 00:21:16
  • vue项目打包后怎样优雅的解决跨域

    2024-04-29 13:11:03
  • go语言beego框架web开发语法笔记示例

    2024-05-21 10:25:22
  • 用Python实现一个简单的线程池

    2023-09-21 15:14:56
  • Python实现学生管理系统并生成exe可执行文件详解流程

    2023-03-11 04:52:42
  • Python装饰器的练习题

    2023-12-07 13:11:48
  • 全面解析Windows下安装 mysql5.7的方法

    2024-01-26 21:06:58
  • mysql 列转行,合并字段的方法(必看)

    2024-01-28 03:09:04
  • Python基于回溯法子集树模板解决马踏棋盘问题示例

    2021-08-01 15:45:43
  • Python学习之线程池与GIL全局锁详解

    2021-10-09 21:55:18
  • Python Django view 两种return的实现方式

    2022-05-03 16:07:04
  • 网页设计应急小技巧

    2011-10-05 18:52:57
  • Golang map如何生成有序的json数据详解

    2024-05-09 09:47:45
  • Django 解决上传文件时,request.FILES为空的问题

    2021-10-21 19:37:41
  • python密码学各种加密模块教程

    2021-03-10 05:32:55
  • 如何把一长串数字分位显示?

    2009-11-06 14:01:00
  • 对Python字符串中的换行符和制表符介绍

    2021-11-11 15:45:29
  • DOM_window对象属性之--clipboardData对象操作代码

    2011-02-05 10:49:00
  • 如何删除vue项目下的node_modules文件夹

    2023-07-02 17:10:00
  • Pyhton中单行和多行注释的使用方法及规范

    2021-11-21 12:13:00
  • asp之家 网络编程 m.aspxhome.com