解析Golang中引用类型是否进行引用传递

 更新时间:2022年08月29日 09:53:15   作者:刘悦的技术博客  
这篇文章主要为大家介绍了Golang中引用类型是否进行引用传递剖析详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

引言

开篇明义,Go lang中从来就不存在所谓的“引用传递”,从来就只有一种变量传递方式,那就是值传递。因为引用传递的前提是存在“引用变量”,但是Go lang中从来就没有出现过所谓的“引用变量”,所以也就不可能存在引用传递这种变量传递的方式。

引用类型

首先,Go lang的基本数据类型是值类型,比如整数、浮点、字符串、布尔、数组及错误类型,它们本质上是原始类型,也就是不可改变的,所以对它们进行操作,一般都会返回一个新创建的值,所以把这些值传递给函数时,其实传递的是一个值的拷贝副本,这一点,基本没啥争议。

而引用类型指的是它的修改动作可以影响到任何引用到它的变量。在 Go 语言中,引用类型有切片(slice)、字典(map)、接口(interface)、函数(func) 以及通道(chan) 。

问题是,如果我们在某一个函数体内对外部定义的引用类型数据做修改操作:

package main  
import "fmt"  
func changeMap(data map[string]string) {  
	data["123"] = "333"  
}  
func main() {  
	a := map[string]string{}  
	a["123"] = "123"  
	fmt.Println("begin:", a)  
	changeMap(a)  
	fmt.Println("after:", a)  
}

程序返回:

begin: map[123:123]  
after: map[123:333]

很明显,函数changeMap改变了外部的字典类型的值,那么我们就可以得出结论,引用类型的传参是使用的引用传递?

引用变量(reference variable)和引用传递(pass-by-reference)

事实上,引用变量(reference variable)和引用传递(pass-by-reference)确实存在,只不过存在于其他的语言中,比如说Python:

a = [2]  
print(id(a))  
def change(a):  
    print(id(a))  
    a.append(1)  
if __name__ == '__main__':  
    print(a)  
    change(a)  
    print(a)

这里我们定义了一个可变数据类型:列表a,然后将它传入函数change中,进行修改操作,同时使用系统内置的id()方法分别打印修改前的值和内存地址以及修改后的值和内存地址,程序返回:

4311179392  
[2]  
4311179392  
[2, 1]

这说明什么?说明变量a是引用变量(reference variable),同时它作为参数的传递方式是引用传递(pass-by-reference),证据就是它原始的内存地址和传递到函数内的内存地址是一致的,都是4311179392。

所以引用变量和引用传递应该具备如下特点:引用变量和原变量的内存地址一样。就像上面的例子里函数内引用变量a和原变量a的内存地址相同。函数使用引用传递,可以改变外部实参的值。就像上面的例子里,change函数使用了引用传递,改变了外部实参a的值。

Golang是否存在引用变量(reference variable)

Go lang中不存在引用变量:

package main  
import "fmt"  
func main() {  
	a := 1  
	var a1 *int = &a  
	var a2 *int = &a  
	fmt.Println("值", a1, " 内存地址:", &a1)  
	fmt.Println("值:", a2, " 内存地址:", &a2)  
}

程序返回:

值 0x140000140b8  内存地址: 0x1400000e028  
值: 0x140000140b8  内存地址: 0x1400000e030

和Python不同的是,在Go lang里,不可能有两个变量有相同的内存地址,所以也就不存在引用变量了。变量a1和a2的值相同,都指向变量a的内存地址,但是变量a1和a2自己本身的内存地址是不一样的,而Python里的引用变量和原变量的内存地址是相同的。

因此,在Go语言里是不存在引用变量的,也就自然没有引用传递了。

字典为什么可以做到值传递但是可以更改原对象?

因为字典虽然名字叫做字典,或者叫做map,但那并不重要,其实它是指针:

package main  
import (  
	"fmt"  
	"unsafe"  
)  
func main() {  
	data := make(map[string]int)  
	var p uintptr  
	fmt.Println("字典大小:", unsafe.Sizeof(data))  
	fmt.Println("指针大小:", unsafe.Sizeof(p))  
}

程序返回:

字典大小: 8  
指针大小: 8

从占据内存空间大小就可以看出,字典和指针其实就是一种东西,那如果字典是指针,那make返回的不应该是*map[string]int吗?为什么我们使用字典传实参,从来都不加*?

在Go lang早期,的确对于字典是使用过指针形式的,但是最后Golang的设计者发现,几乎没有人使用字典不加指针,因此就直接去掉了形式上的指针符号*,类比的话,我们会发现现实中几乎从来就没有人管AC米兰叫AC米兰,都是直呼米兰,因为大家都认为米兰就是AC米兰,所以都自动省略了形式上的“AC”。

本质上,我们可以理解字典作为参数传递方式是值传递,只不过引用类型传递的是一个指向底层数据的指针,所以我们在操作的时候,可以修改共享的底层数据的值,进而影响到所有引用到这个共享底层数据的变量,这也就是为什么字典在函数内操作可以影响原对象的原因。

结语

引用类型之所以可以引用,是因为我们创建引用类型的变量,其实是一个标头值,标头值里包含一个指针,指向底层的数据结构,当我们在函数中传递引用类型时,其实传递的是这个标头值的副本,它所指向的底层结构并没有被复制传递,这也是引用类型传递高效的原因,换句话说,Go lang为了保证值传递的纯粹性,才引入了指针的概念,如果Go lang里存在引用变量和引用传递,那指针不就成了画蛇添足的浮笔浪墨了吗?

以上就是解析Golang中引用类型是否进行引用传递的详细内容,更多关于Go 引用类型引用传递的资料请关注脚本之家其它相关文章!

相关文章

  • Golang Printf,Sprintf,Fprintf 格式化详解

    Golang Printf,Sprintf,Fprintf 格式化详解

    这篇文章主要介绍了Golang Printf,Sprintf,Fprintf 格式化详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-03-03
  • golang gorm多条件筛选查询操作

    golang gorm多条件筛选查询操作

    这篇文章主要介绍了golang gorm多条件筛选查询操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • Golang import本地包和导入问题相关详解

    Golang import本地包和导入问题相关详解

    这篇文章主要介绍了Golang import本地包和导入问题相关详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-02-02
  • 深入了解Golang中的数据类型

    深入了解Golang中的数据类型

    在计算机编程中,数据类型是非常重要的一个概念。这篇文章将详细介绍 Golang中的数据类型,包括基本类型、复合类型、引用类型以及自定义类型,希望对大家有所帮助
    2023-04-04
  • golang中channel+error来做异步错误处理有多香

    golang中channel+error来做异步错误处理有多香

    官方推荐golang中错误处理当做值处理, 既然是值那就可以在channel中传输,这篇文章主要介绍了golang 错误处理channel+error真的香,需要的朋友可以参考下
    2023-01-01
  • 使用go语言解析xml的实现方法(必看篇)

    使用go语言解析xml的实现方法(必看篇)

    下面小编就为大家带来一篇使用go语言解析xml的实现方法(必看篇)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-06-06
  • 如何在Golang中运行JavaScript

    如何在Golang中运行JavaScript

    最近写一个程序,接口返回的数据是js格式的,需要通过golang来解析js,所以下面这篇文章主要给大家介绍了关于如何在Golang中运行JavaScript的相关资料,需要的朋友可以参考下
    2022-01-01
  • Golang RPC的原理与简单调用详解

    Golang RPC的原理与简单调用详解

    RPC(Remote Procedure Call),主要是帮助我们屏蔽网络编程细节 ,使我们更专注于业务逻辑,所以本文主要来和大家聊聊RPC的原理与简单调用,希望对大家有所帮助
    2023-05-05
  • Go语言Gin框架获取请求参数的两种方式

    Go语言Gin框架获取请求参数的两种方式

    在添加路由处理函数之后,就可以在路由处理函数中编写业务处理代码了,而编写业务代码第一件事一般就是获取HTTP请求的参数吧,Gin框架在net/http包的基础上封装了获取参数的方式,本文小编给大家介绍了获取参数的两种方式,需要的朋友可以参考下
    2024-01-01
  • go高并发时append方法偶现错误解决分析

    go高并发时append方法偶现错误解决分析

    这篇文章主要为大家介绍了go高并发时append方法偶现错误解决分析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-10-10

最新评论