其实go的参数传递,核心就是一句话:go里所有参数传递都是值传递,既把参数复制一份放到函数里去用。
go的函数传参,不管参数是什么类型,都会复制一份,然后新的参数在函数内部被使用。
不像其他语言,有的时候传参只是传递一个原来参数的引用(引用和指针的区别,欢迎翻看我上上上上一篇分享),在函数内部操作变量,其实还是操作的原变量。go内不会直接的操作原变量。
关于指针相比于C里的指针,go内部的指针一个被简化过的指针,指针可以取值获取其变量;变量可以取地址获取一个指针类型的值。 但是不可以对指针执行 地址的加减操作(unsafe.Pointer 可以,不在本次讨论范围之内)。
我觉得这个简化挺好,保留了参数传递时避免大变量的优势,又去掉了复杂性。
下面来通过实例具体说明之所以用 切片做示例,是因为 切片是引用类型,也就是说切片内部有一个指针 指向底层放数据的数组。 类似于 一个指针变量。由于它的这个特性,更能说明问题。
错误示例1 package main import "log" var str []string func main() { setVal(str) log.Println(str) } //需要在这里赋值str,但是又不能直接引用 str func setVal(val []string) { val = []string{"a", "b"} }结果是空数组 虽然切片是引用类型,还是没有复制成功。 原因就是,在传参之前 这个切片并没有被赋值,它内部的指针是一个空的。
传参的时候,复制了一个切片变量,这个新的变量 指向setVal函数内的实例变量。但是setVal 内的操作并不影响原来的切片变量。
错误示例2 package main import "log" var str *[]string func main() { setVal(str) log.Println(str) } //需要在这里赋值str,但是又不能直接引用 str func setVal(val *[]string) { val = &[]string{"a", "b"} }这个例子,看似使用了指针,不仔细就得话,可能觉得指针作为参数应该会对原参数产生效果,但其实不会,最后的结果是 nil。
其错误原因与上一个例子基本一致,虽然传递的是一个 指针。但是 这个指针只是原参数的复制品,一个新的指针,由于参数传递时,指针并没有复制,这个新的指针跟原来的指针毫无关系。
正确版本: package main import "log" var str []string func main() { setVal(&str) log.Println(str) } //需要在这里赋值str,但是又不能直接引用 str func setVal(val *[]string) { *val = []string{"a", "b"} }注意看main 函数,这个时候原参数是一个变量,main 函数内作为参数的是这个变量的地址,同时也是一个指针变量。在setVal内复制了一个指针变量,其值同样是这个原参数的地址。所以这个例子里setVal 函数内的形参指针进行取值, 会取到原参数,所以其操作会对原参数产生影响。
总结没啥好总结的了,总结都被写在前边了,还总结个毛线 ~ ~