Go中数组传参的几种方式小结

2023-03-08 11:03:46 数组 几种 小结

初学golang数组传参问题就是把我整不会了,以前我们使用C语言进行数组传参时是这样传递的,直接传递数组的起始地址即可。

而在Go中数组传参我们以以下两种方式进行传递,这里我直接贴出go代码,文中有注释:

package main

import "fmt"

func testArr01(arr []int) { //形参未指定大小
	arr[0]++
}
func testArr02(arr [5]int) { //形参指定大小
	arr[0]++
}
func testArr03(arr *[5]int) { //使用指针方式,也就是引用传递
	arr[0]++
}
func main() {
	var arr = [5]int{0, 2, 3, 17, 50}
	testArr01(arr[:]) //切片方式传递(常用)
	for i := 0; i < len(arr); i++ {
		fmt.Printf("%d ", arr[i])
	}
	fmt.Println()

	testArr02(arr) //数组名方式传递,此时传递的是副本,并不会改变原数组
	for i := 0; i < len(arr); i++ {
		fmt.Printf("%d ", arr[i])
	}
	fmt.Println()

	testArr03(&arr) //指针方式传递,会改变原数组
	for i := 0; i < len(arr); i++ {
		fmt.Printf("%d ", arr[i])
	}
	fmt.Println()
}


运行结果:

至此,我们发现,通过数组名传递参数方式并不能修改原数组,而通过切片方式传递能完成数组修改,这是因为,和其他语言不同,go语言在将数组名作为函数参数的时候,参数传递即是对数组的复制。在形参中对数组元素的修改都不会影响到数组元素原来的值。而在使用slice作为函数参数时,进行参数传递将是一个地址拷贝,即将底层数组的内存地址复制给参数slice。这时,对slice元素的操作就是对底层数组元素的操作。采用指针方式进行传递就是传递的引用,对这个引用指向的地址的内容进行修改也会影响原数组的值。

总结:在Go语言中函数的参数有两种传递方式,按值传递和按引用传递。Go默认使用按值传递来传递参数,也就是传递参数的副本。在函数中对副本的值进行更改操作时,不会影响到原来的变量。按引用传递其实也可以称作"按值传递",只不过该副本是一个地址的拷贝,通过它可以修改这个值所指向的地址上的值。

Go语言中,在函数调用时,引用类型(slice、map、interface、channel)都默认使用引用传递。

数组传递时的一些缺点

一般情况下,传递指针的消耗比传递副本的少,尤其是当数组特别大时。具体原因是:

  • 值传递需要完整的复制初始数组并将这份拷贝放到栈中,这将耗费大量运行时间,因而值传递方式的效率比较低。
  • 初始数组的拷贝需要占用额外的内存空间(栈中的内存)
  • 编译程序需要专门产生一部分用来复制初始数组的代码,这将使程序变大。

相关文章