共计 854 个字符,预计需要花费 3 分钟才能阅读完成。
前情提要
最近看很多教程或者说博客上都说 golang 中的 slice、map、channel、func
都是“援用传递”
,然而一方面又说 golang 中所有类型都是值传递,总感觉有些云里雾里的,于是我亲自做了下测试和思考。
这里是代码局部:
package main
import ("fmt")
func test(a *int) {fmt.Println("传入变量的值:", a)
fmt.Println("传入变量的地址:", &a)
}
func main() {
va := 666
vad := &va
fmt.Println("须要传入的值:", vad)
fmt.Println("须要传入的值的地址", &vad)
test(vad)
}
这里是执行后果
须要传入的值:0xc000018658
须要传入的值的地址 0xc000006058
传入变量的值: 0xc000018658
传入变量的地址:0xc000006060
思考讲解
也就是说传入和理论接管的值都是指针变量,这个两个指针变量 vad
和 a
的值都为指针所指向的变量 va
的地址 0xc000018658
。
而后再看函数外部的这个传入的这个指针 a
的地址(指针)0xc000006060
,比照里面寄存指针 vad
的地址 0xc000006058
,这两个值是不一样的,阐明指针类型也是值传递,也就是说复制了一份指针的值传递给函数。
所以来说,函数 test
外部的 a
变量和内部的 vad
变量齐全不是同一个货色,a
是 vad
的复制体,然而这两个变量的值寄存的都是 va
变量的地址,所以操作 a
会对变量 va
产生批改。
从这里来看,集体感觉“slice、map、channel、func
都是援用传递”的表述形式感觉容易引起误会,会狐疑 golang
的设计对这几个货色非凡看待,是援用传递。
实际上 golang
的设计,所有类型都是以 值
的模式传递。只不过对这几种类型来说,底层的实现就是这几种类型的数据创立胜利后,变量所接管的数据是这些类型所对应的地址,或者说被赋值的变量所承受到的是这几种类型的值的地址。而不应该是这几种类型在传递的时候是什么援用类型。
正文完