我的编程空间,编程开发者的网络收藏夹
学习永远不晚

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

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

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

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

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

1、Go 官方的定义

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

如同 C 系列的所有语言一样,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 语言确实都是值传递。那是不是在函数内修改值,就不会影响到 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 语言只有值传递,

到此这篇关于关于Go 是传值还是传引用?的文章就介绍到这了,更多相关Go 是传值还是传引用内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

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

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Go语言参数传递是传值还是传引用

目录什么是传值(值传递)什么是传引用(引用传递)迷惑Mapchan类型和map、chan都不一样的slice小结对于了解一门语言来说,会关心我们在函数调用的时候,参数到底是传的值,还是引用? 其实对于传值和传引用,是一个比较古老的话题,做研
2022-06-07

Java语言是传值还是传引用

小编给大家分享一下Java语言是传值还是传引用,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1. 简单类型是按值传递的  Java 方法的参数是简单类型的时候,是
2023-06-03

Java是值传递还是引用传递

本篇内容主要讲解“Java是值传递还是引用传递”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java是值传递还是引用传递”吧!1.值类型通俗意义上来说,所谓的值类型指的就是 Java 中的 8
2023-06-16

Java编程是值传递还是引用传递

这篇文章主要介绍“Java编程是值传递还是引用传递”,在日常操作中,相信很多人在Java编程是值传递还是引用传递问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java编程是值传递还是引用传递”的疑惑有所帮助!
2023-06-30

php数组是传值还是引用

php数组是传值。PHP数组传递是值传递;在调用函数时通过将PHP数组作为实参赋给形参,在函数中修改,并不会影响到数组本身,说明此过程中的传递是值传递,数组变量并非指向此数组本身的引用。
2023-05-14

java支持值传递还是引用传递

本篇内容主要讲解“java支持值传递还是引用传递”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“java支持值传递还是引用传递”吧!文章目的:验证Java语言到底是值传递还是引用传递以及Java参
2023-06-30

java经典问题:传值还是传引用(转)

java经典问题:传值还是传引用(转)[@more@]经典的问题,但却不容易弄懂,尤其对有c基础的java程序员来说,更容易引起混乱,这里我试图简单点描述。 “java函数是传值的,java函数传递的参数是对象的引用” 这两句话好像初听上去
2023-06-03

一文搞懂Golang 值传递还是引用传递

最多人犯迷糊的就是 slice、map、chan 等类型,都会认为是 “引用传递”,从而认为 Go 语言的 xxx 就是引用传递。正因为它们还引用类型(指针、map、slice、chan等这些),这样就可以修改原内容数据,这篇文章主要介绍了Golang 值传递还是引用传递,需要的朋友可以参考下
2023-01-11

深入理解python中函数传递参数是值传递还是引用传递

目前网络上大部分博客的结论都是这样的: Python不允许程序员选择采用传值还是传 引用。Python参数传递采用的肯定是“传对象引用”的方式。实际上,这种方式相当于传值和传引用的一种综合。如果函数收到的是一个可变对象(比如字典 或者列表)
2022-06-04

Java的传值与传引用是什么

本篇内容介绍了“Java的传值与传引用是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!通常的说法是:对于基本数据类型(整型、浮点型、字符
2023-06-17

C++中传值、传地址和传引用的区别是什么

小编给大家分享一下C++中传值、传地址和传引用的区别是什么,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!传引用定义传值与传地址,相信大家都了如指掌了,在这里先介绍
2023-06-20

go语言中是否引用传递

这篇文章主要讲解了“go语言中是否引用传递”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“go语言中是否引用传递”吧!没有。Go没有引用变量,所以Go语言里调用函数的时也没有引用传递。Go程序
2023-07-04

深入了解Go语言的传值和传引用

go 语言中传值和传引用有两种传递机制:传值:传递参数值的副本,对副本的更改不会影响原始值。传引用:传递参数值的引用,允许函数或方法修改原始值。了解这两种机制对于编写高效、可维护的代码至关重要,因为它们会影响代码的行为和效率。深入了解 Go
深入了解Go语言的传值和传引用
2024-04-04

php按值传递和引用传递的区别是什么

这篇文章主要讲解了“php按值传递和引用传递的区别是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“php按值传递和引用传递的区别是什么”吧!说明1、按值传递中php必须复制值。特别是对于
2023-06-20

java中值传递和引用传递的区别是什么

在Java中,值传递(pass by value)和引用传递(pass by reference)是两种不同的参数传递方式。值传递是指当将一个变量作为参数传递给方法时,传递的是变量的值而不是变量本身。在方法内部对参数进行修改不会影响原始变量
2023-08-14

在 Go to 函数中通过引用和值传递

在PHP开发中,Go to 函数是一个非常常用的函数,可以用于控制程序的流程。在使用Go to函数时,我们有两种传递参数的方式:通过引用和通过值传递。通过引用传递参数时,函数内部对参数的修改会影响到函数外部的变量值。而通过值传递参数时,函数
在 Go to 函数中通过引用和值传递
2024-02-13

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录