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

Go语言中值接收者和指针接收者的区别?

在 Go 语言中,值接收者指针接收者是方法定义中的两种接收者类型。它们的主要区别在于方法调用时的行为、接收者是否可以被修改,以及性能上的差异。


值接收者

定义

值接收者的方法接收的是调用对象的一个副本,方法内部对该副本的修改不会影响原对象。

特点

  1. 传递的是值的副本
    • 方法内操作的是值的拷贝,修改不会影响原对象。
  2. 适用于不可变的对象
    • 如果方法不需要修改对象的状态,可以使用值接收者。
  3. 适用于小对象
    • 值接收者传递整个对象副本,适合较小的对象。
  4. 可用值或指针调用
    • 无论是值还是指针,都可以调用值接收者的方法,Go 会自动处理。

示例

package main

import "fmt"

type Rectangle struct {
    Width, Height int
}

// 值接收者方法
func (r Rectangle) Area() int {
    return r.Width * r.Height
}

// 修改操作(仅对副本有效)
func (r Rectangle) SetWidth(w int) {
    r.Width = w
}

func main() {
    rect := Rectangle{Width: 10, Height: 5}
    fmt.Println("Area:", rect.Area()) // 输出: Area: 50

    rect.SetWidth(20)
    fmt.Println("Width after SetWidth:", rect.Width) // 输出: Width after SetWidth: 10
}

说明:调用 SetWidth 方法时,rect.Width 没有改变,因为 SetWidth 操作的是值的副本。


指针接收者

定义

指针接收者的方法接收的是调用对象的地址,方法内部对接收者的修改会直接作用于原对象。

特点

  1. 传递的是指针
    • 方法操作的是对象的指针,修改接收者的状态会直接影响原对象。
  2. 适用于可变的对象
    • 如果方法需要修改对象的状态,必须使用指针接收者。
  3. 适用于大对象
    • 指针接收者避免了拷贝大对象的开销。
  4. 只能用值或指针调用
    • 无论对象是值还是指针,都可以调用指针接收者的方法,Go 会自动转换。

示例

package main

import "fmt"

type Rectangle struct {
    Width, Height int
}

// 指针接收者方法
func (r *Rectangle) Scale(factor int) {
    r.Width *= factor
    r.Height *= factor
}

func main() {
    rect := Rectangle{Width: 10, Height: 5}
    rect.Scale(2)
    fmt.Println("Scaled Width:", rect.Width) // 输出: Scaled Width: 20
    fmt.Println("Scaled Height:", rect.Height) // 输出: Scaled Height: 10
}

说明:调用 Scale 方法时,rect.Widthrect.Height 的值被直接修改。


值接收者与指针接收者的主要区别

特性值接收者指针接收者
传递内容对象的副本对象的地址
方法是否能修改原对象
适用场景- 不需要修改接收者
- 对象较小,拷贝开销小
- 需要修改接收者
- 对象较大,拷贝开销大
调用方式值或指针都可以调用值或指针都可以调用
性能较低性能:对于大对象,会复制整个对象较高性能:传递指针,避免了对象的复制

值接收者与指针接收者的调用规则

无论方法是值接收者还是指针接收者:

  1. 值调用值接收者:直接使用。
  2. 指针调用值接收者:Go 会自动解引用指针并传递对象值的副本。
  3. 值调用指针接收者:Go 会自动获取对象的地址。
  4. 指针调用指针接收者:直接使用。

示例

package main

import "fmt"

type Counter struct {
    Count int
}

func (c Counter) Increment() {
    c.Count++
}

func (c *Counter) Decrement() {
    c.Count--
}

func main() {
    c := Counter{Count: 10}

    // 调用值接收者方法
    c.Increment()
    fmt.Println("After Increment:", c.Count) // 输出: After Increment: 10 (未修改)

    // 调用指针接收者方法
    c.Decrement()
    fmt.Println("After Decrement:", c.Count) // 输出: After Decrement: 9 (被修改)
}

如何选择值接收者或指针接收者?

  • 选择值接收者

    • 方法不需要修改接收者。
    • 对象较小(如基本数据类型或简单结构体)。
    • 提升代码可读性。
  • 选择指针接收者

    • 方法需要修改接收者。
    • 对象较大,拷贝成本高。
    • 需要保证方法在所有调用场景中的行为一致(如使用接口时,避免副本调用导致的问题)。

示例

如果结构体实现接口时:

  • 建议使用指针接收者,以确保通过接口调用时能修改对象。
type Resizable interface {
    Resize(factor int)
}

type Rectangle struct {
    Width, Height int
}

func (r *Rectangle) Resize(factor int) {
    r.Width *= factor
    r.Height *= factor
}

总结

  • 值接收者适合不可变操作;指针接收者适合可变操作。
  • 指针接收者避免了拷贝大对象的开销,但增加了代码复杂性。
  • 选择接收者类型时,应根据方法的用途和对象的大小权衡。

http://www.kler.cn/a/461381.html

相关文章:

  • 实时数仓与离线数仓的全面对比
  • 简单使用linux
  • redux react-redux @reduxjs/toolkit
  • df.replace({‘b‘: r‘\s*(\.)\s*‘}, {‘b‘: r‘\1ty‘}, regex=True)
  • 使用Docker部署最新版JupyterHub
  • [最佳方法] 如何将视频从 Android 发送到 iPhone
  • HTML<select>标签有关的定义和属性
  • 【人工智能机器学习基础篇】——深入详解监督学习之模型评估:掌握评估指标(准确率、精确率、召回率、F1分数等)和交叉验证技术
  • c# Record关键字
  • Github 正常访问但是ping不同也无法进行git操作
  • 通过无障碍服务(AccessibilityService)实现Android设备全局水印显示
  • Docker 搭建 Gogs
  • SpringBoot 实现登录功能
  • 书生·浦语大模型全链路开源体系-第9关 LMDeploy 量化部署进阶实践
  • TB1801D 线性驱动 LED 恒流芯片
  • 苹果系统MacOS下采用ObjectC访问opencv加载图片的一个简单实例
  • Flink的多流转换(分流-侧输出流、合流-union、connect、join)
  • 中华人民共和国网络安全法
  • BOE(京东方)“向新2025”年终媒体智享会落地深圳
  • 关于MCU复位电路的分析与设计
  • Java重要面试名词整理(十二):Netty
  • C++ Brain Teasers: 未指定和实现定义的行为-函数参数的求值顺序
  • 网络安全靶场合集:知识点与功能解析
  • 数据可视化-1:使用Matplotlib绘制多种图表
  • 手机租赁平台开发助力智能设备租赁新模式
  • 机器学习、深度学习、神经网络之间的关系