在函数的最后一个参数的数据类型之前加上省略号 ...
,表示该参数的数据类型是 变长类型
, 调用该函数时可以传递任意数量 ( 0 - N )
的该类型的参数。
一个函数只能有一个变长参数,且变长参数必须为最后一个参数。
package main import "fmt" func sum(numbers ...int) int { total := 0 for _, num := range numbers { total += num } return total } func main() { fmt.Printf("1 = %d\n", sum(1)) } // $ go run main.go // 输出如下 /** 1 = 1 */
package main import "fmt" func sum(numbers ...int) int { total := 0 for _, num := range numbers { total += num } return total } func main() { fmt.Printf("1 + 2 + 3 = %d\n", sum(1, 2, 3)) } // $ go run main.go // 输出如下 /** 1 + 2 + 3 = 6 */
如果要传递 切片类型
数据作为函数参数,只需要在参数后面加上省略号 ...
。
package main import "fmt" func sum(numbers ...int) int { total := 0 for _, num := range numbers { total += num } return total } func main() { numbers := []int{1, 2, 3} fmt.Printf("1 + 2 + 3 = %d\n", sum(numbers...)) // 切片变量后面加 ... 即可 } // $ go run main.go // 输出如下 /** 1 + 2 + 3 = 6 */
package main import "fmt" func sum(numbers ...int) int { total := 0 for _, num := range numbers { total += num } return total } func main() { fmt.Printf("不传递任何参数 = %d\n", sum()) } // $ go run main.go // 输出如下 /** 不传递任何参数 = 0 */
建议先阅读指针小节。
package main import "fmt" func double(n int) { n *= 2 } func doubleWithPtr(n *int) { *n *= 2 } func main() { n := 100 double(n) fmt.Printf("使用普通变量作为函数参数执行完成后, n = %d\n", n) // 可以看到,变量值并未发生变化 doubleWithPtr(&n) fmt.Printf("使用指针变量作为函数参数执行完成后, n = %d\n", n) // 变量值已经发生变化 } // $ go run main.go // 输出如下 /** 使用普通变量作为函数参数执行完成后, n = 100 使用指针变量作为函数参数执行完成后, n = 200 */
package main import "fmt" func double(numbers [3]int) { for _, v := range numbers { v *= 2 } } func doubleWithPtr(numbers *[3]int) { for i := range numbers { numbers[i] *= 2 } } func main() { numbers := [3]int{100, 200, 300} double(numbers) fmt.Printf("使用数组变量作为函数参数执行完成后, n = %d\n", numbers) // 可以看到,数组元素并未发生变化 doubleWithPtr(&numbers) fmt.Printf("使用指针数组变量作为函数参数执行完成后, n = %d\n", numbers) // 数组元素已经发生变化 } // $ go run main.go // 输出如下 /** 使用数组变量作为函数参数执行完成后, n = [100 200 300] 使用指针数组变量作为函数参数执行完成后, n = [200 400 600] */
在 切片 小节中讲到,切片的底层引用了一个数组,可以简单地理解为:切片本身是一个指针,指向底层数组的元素, 所以常用的方式的是将函数参数定义为切片类型。
package main import "fmt" func double(numbers []int) { for i := range numbers { numbers[i] *= 2 } } func main() { numbers := []int{100, 200, 300} double(numbers) fmt.Printf("使用切片变量作为函数参数执行完成后, n = %d\n", numbers) // 切片元素已经发生变化 } // $ go run main.go // 输出如下 /** 使用切片变量作为函数参数执行完成后, n = [200 400 600] */
默认情况下,Go 的函数参数传递方式使用 值传递
,也就是说,修改参数的值不会影响到 原变量
, 如果希望修改参数的同时修改 原变量
,需要传递变量的引用 (指针)。
对于基础数据类型 (如 int, float, bool 等),需要传递值的地址,对于已经是 引用类型
的数据类型 (如切片、通道等),直接传递值即可。