关于Go 是传值还是传引用?

作者:煎鱼 时间:2024-04-26 17:26:38 

目录
  • 1、Go 官方的定义

  • 2、传值和传引用

    • 2.1 传值

    • 2.2 传引用

  • 3、争议最大的 map 和 slice

    • 3.1 map

    • 3.2 slice

  • 3、总结

    关于Go 是传值还是传引用?很多人都讨论起来

    关于Go 是传值还是传引用?

    下面我们就带着问题一起探索答案吧

    1、Go 官方的定义

    本部分引用 Go 官方 FAQ 的 “When are function parameters passed by value?”,内容如下。

    如同 C 系列的所有语言一样,Go 语言中的所有东西都是以值传递的。也就是说,一个函数总是得到一个被传递的东西的副本,就像有一个赋值语句将值赋给参数一样。

    关于Go 是传值还是传引用?

    例如:

    • 向一个函数传递一个 int 值,就会得到 int 的副本。而传递一个指针值就会得到指针的副本,但不会得到它所指向的数据。

    • map slice 的行为类似于指针:它们是包含指向底层 map slice 数据的指针的描述符。

    • 复制一个 map slice 值并不会复制它所指向的数据。

    • 复制一个接口值会复制存储在接口值中的东西。

    • 如果接口值持有一个结构,复制接口值就会复制该结构。如果接口值持有一个指针,复制接口值会复制该指针,但同样不会复制它所指向的数据。

    划重点:Go 语言中一切都是值传递,没有引用传递。不要直接把其他概念硬套上来,会犯先入为主的错误的。

    2、传值和传引用

    2.1 传值

    传值,也叫做值传递(pass by value)。其指的是在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。

    简单来讲,值传递,所传递的是该参数的副本,是复制了一份的,本质上不能认为是一个东西,指向的不是一个内存地址。

    案例一如下:


    func main() {
    s := "脑子进煎鱼了"
    fmt.Printf("main 内存地址:%p\n", &s)
    hello(&s)
    }

    func hello(s *string) {
    fmt.Printf("hello 内存地址:%p\n", &s)
    }

    输出结果:

    main 内存地址:0xc000116220
    hello 内存地址:0xc000132020

    我们可以看到在 main 函数中的变量 s 所指向的内存地址是 0xc000116220。在经过 hello 函数的参数传递后,其在内部所输出的内存地址是 0xc000132020,两者发生了改变。

    关于Go 是传值还是传引用?

    据此我们可以得出结论,在 Go 语言确实都是值传递。那是不是在函数内修改值,就不会影响到 main 函数呢?

    案例二如下:


    func main() {
    s := "脑子进煎鱼了"
    fmt.Printf("main 内存地址:%p\n", &s)
    hello(&s)
    fmt.Println(s)
    }

    func hello(s *string) {
    fmt.Printf("hello 内存地址:%p\n", &s)
    *s = "煎鱼进脑子了"
    }

    我们在 hello 函数中修改了变量 s 的值,那么最后在 main 函数中我们所输出的变量 s 的值是什么呢。是 “脑子进煎鱼了”,还是 "煎鱼进脑子了"?

    输出结果:

    main 内存地址:0xc000010240
    hello 内存地址:0xc00000e030
    煎鱼进脑子了

    输出的结果是 “煎鱼进脑子了”。这时候大家可能又犯嘀咕了,煎鱼前面明明说的是 Go 语言只有值传递,也验证了两者的内存地址,都是不一样的,怎么他这下他的值就改变了,这是为什么?

    因为 “如果传过去的值是指向内存空间的地址,那么是可以对这块内存空间做修改的”。

    也就是这两个内存地址,其实是指针的指针,其根源都指向着同一个指针,也就是指向着变量 s。因此我们进一步修改变量 s,得到输出 “煎鱼进脑子了” 的结果。

    2.2 传引用

    传引用,也叫做引用传递(pass by reference),指在调用函数时将实际参数的地址直接传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。

    在 Go 语言中,官方已经明确了没有传引用,也就是没有引用传递这一情况。

    因此借用文字简单描述,像是例子中,即使你将参数传入,最终所输出的内存地址都是一样的。

    3、争议最大的 map 和 slice

    这时候又有小伙伴疑惑了,你看 Go 语言中的 map slice 类型,能直接修改,难道不是同个内存地址,不是引用了?

    其实在 FAQ 中有一句提醒很重要:“map slice 的行为类似于指针,它们是包含指向底层 map slice 数据的指针的描述符”。

    3.1 map

    针对 map 类型,进一步展开来看看例子:


    func main() {
    m := make(map[string]string)
    m["脑子进煎鱼了"] = "这次一定!"
    fmt.Printf("main 内存地址:%p\n", &m)
    hello(m)

    fmt.Printf("%v", m)
    }

    func hello(p map[string]string) {
    fmt.Printf("hello 内存地址:%p\n", &p)
    p["脑子进煎鱼了"] = "记得点赞!"
    }

    输出结果:

    main 内存地址:0xc00000e028
    hello 内存地址:0xc00000e038

    确实是值传递,那修改后的 map 的结果应该是什么。既然是值传递,那肯定就是 "这次一定!",对吗?

    输出结果:

    map[脑子进煎鱼了:记得点赞!]

    结果是修改成功,输出了 “记得点赞!”。这下就尴尬了,为什么是值传递,又还能做到类似引用的效果,能修改到源值呢?

    这里的小窍门是:


    func makemap(t *maptype, hint int, h *hmap) *hmap {}

    这是创建 map 类型的底层 runtime 方法,注意其返回的是 *hmap 类型,是一个指针。也就是 Go 语言通过对 map 类型的相关方法进行封装,达到了用户需要关注指针传递的作用。

    就是说当我们在调用 hello 方法时,其相当于是在传入一个指针参数 hello(*hmap),与前面的值类型的案例二类似。

    这类情况我们称其为 “引用类型”,但 “引用类型” 不等同于就是传引用,又或是引用传递了,还是有比较明确的区别的。

    在 Go 语言中与 map 类型类似的还有 chan 类型:


    func makechan(t *chantype, size int) *hchan {}

    一样的效果。

    3.2 slice

    针对 slice 类型,进一步展开来看看例子:


    func main() {
    s := []string{"烤鱼", "咸鱼", "摸鱼"}
    fmt.Printf("main 内存地址:%p\n", s)
    hello(s)
    fmt.Println(s)
    }

    func hello(s []string) {
    fmt.Printf("hello 内存地址:%p\n", s)
    s[0] = "煎鱼"
    }

    输出结果:

    main 内存地址:0xc000098180
    hello 内存地址:0xc000098180
    [煎鱼 咸鱼 摸鱼]

    从结果来看,两者的内存地址一样,也成功的变更到了变量 s 的值。这难道不是引用传递吗,煎鱼翻车了?

    关注两个细节:

    • 没有用 & 来取地址。

    • 可以直接用 %p 来打印。

    之所以可以同时做到上面这两件事,是因为标准库 fmt 针对在这一块做了优化:


    func (p *pp) fmtPointer(value reflect.Value, verb rune) {
    var u uintptr
    switch value.Kind() {
    case reflect.Chan, reflect.Func, reflect.Map, reflect.Ptr, reflect.Slice, reflect.UnsafePointer:
     u = value.Pointer()
    default:
     p.badVerb(verb)
     return
    }

    留意到代码 value.Pointer,标准库进行了特殊处理,直接对应的值的指针地址,当然就不需要取地址符了。

    标准库 fmt 能够输出 slice 类型对应的值的原因也在此:


    func (v Value) Pointer() uintptr {
    ...
    case Slice:
     return (*SliceHeader)(v.ptr).Data
    }
    }

    type SliceHeader struct {
    Data uintptr
    Len  int
    Cap  int
    }

    其在内部转换的 Data 属性,正正是 Go 语言中 slice 类型的运行时表现 SliceHeader。我们在调用 %p 输出时,是在输出 slice 的底层存储数组元素的地址。

    下一个问题是:为什么 slice 类型可以直接修改源数据的值呢。

    其实和输出的原理是一样的,在 Go 语言运行时,传递的也是相应 slice 类型的底层数组的指针,但需要注意,其使用的是指针的副本。严格意义是引用类型,依旧是值传递。

    妙不妙?

    3、总结

    在今天这篇文章中,我们针对 Go 语言的日经问题:“Go 语言到底是传值(值传递),还是传引用(引用传递)” 进行了基本的讲解和分析。

    另外在业内中,最多人犯迷糊的就是 slicemapchan 等类型,都会认为是 “引用传递”,从而认为 Go 语言的 xxx 就是引用传递,我们对此也进行了案例演示。

    这实则是不大对的认知,因为:“如果传过去的值是指向内存空间的地址,是可以对这块内存空间做修改的”。

    其确实复制了一个副本,但他也借由各手段(其实就是传指针),达到了能修改源数据的效果,是引用类型。

    石锤,Go 语言只有值传递,

    来源:https://juejin.cn/post/7018403965712744478

    标签:Go,传值,传引用
    0
    投稿

    猜你喜欢

  • keras自动编码器实现系列之卷积自动编码器操作

    2023-12-31 18:33:15
  • python socket网络编程步骤详解(socket套接字使用)

    2022-09-15 11:35:08
  • JS基于面向对象实现的选项卡效果示例

    2024-04-19 10:42:56
  • python判断字符串是否包含子字符串的方法

    2021-01-04 12:48:03
  • Python RuntimeError: thread.__init__() not called解决方法

    2022-12-22 17:11:46
  • Python使用背景差分器实现运动物体检测

    2023-06-21 09:37:22
  • php中替换字符串函数strtr()和str_repalce()的用法与区别

    2023-11-17 06:12:53
  • 关注前端开发流程

    2009-12-31 17:38:00
  • php生成shtml类用法实例

    2023-11-15 21:46:23
  • javascript管中窥豹 形参与实参浅析

    2024-04-16 09:25:54
  • python持久化存储文件操作方法

    2022-06-10 00:59:34
  • Thinkphp5框架实现获取数据库数据到视图的方法

    2024-05-03 15:52:16
  • 微软建议的ASP性能优化28条守则(8)

    2005-05-30 16:04:00
  • 利用Python的tkinter模块实现界面化的批量修改文件名

    2023-08-30 20:45:15
  • 扫盲大讲堂:mysql出错的代码解析及解答

    2009-09-05 10:08:00
  • Django REST为文件属性输出完整URL的方法

    2023-07-29 02:42:42
  • Appium+Python实现简单的自动化登录测试的实现

    2021-09-13 05:49:14
  • python面向对象入门教程之从代码复用开始(一)

    2022-07-17 21:54:13
  • vue iview的菜单组件Mune 点击不高亮的解决方案

    2024-06-07 15:20:08
  • golang实现跨域访问的方法

    2024-02-15 18:33:31
  • asp之家 网络编程 m.aspxhome.com