爱上开源之golang入门至实战第四章函数(Func)(五)
4.4.5 变长参数
变长参数传递的函数在很多语言里都是支持的,javascript里就有这样的函数, java在1.5以后也开始支持, 同样在go语言里也支持这样非常方便的变长参数的函数。 比如,我们最常用到的fmt.Println(), 见print.go源文件
// Println formats using the default formats for its operands and writes to standard output.
// Spaces are always added between operands and a newline is appended.
// It returns the number of bytes written and any write error encountered.
func Println(a ...any) (n int, err error) {
return Fprintln(os.Stdout, a...)
}
如上段源代码,Println函数就是变长参数的函数, ...any,表示数据类型为any的变参数;
下面我们示例一段变长参数的函数的例子
func Average(n ...int) float64 {
if len(n) == 0 {
return 0
}
sum := 0.0
for _, v := range n {
sum += float64(v)
}
return sum / float64(len(n)*1.0)
}
fmt.Printf("Average()=%v \n", Average())
fmt.Printf("Average(1)=%v \n", Average(1))
fmt.Printf("Average(1, 6)=%v \n", Average(1, 6))
fmt.Printf("Average(1, 2, 3, 4, 5)=%v \n", Average(1, 2, 3, 4, 5))
n := append(make([]int, 0), 2, 4, 6, 8, 10, 12)
fmt.Printf("Average(%v)=%v \n", n, Average(n...))
==== OUTPUT ====
Average()=0
Average(1)=1
Average(1, 6)=3.5
Average(1, 2, 3, 4, 5)=3
Average([2 4 6 8 10 12])=7
变长函数的最后一个参数是采用 ...type 的形式,那么这个函数就可以处理一个变长的参数,这个长度可以为 0,这样的函数称为变参函数。如上个例子,我们就定义了一个Average的变长参数的函数,Average函数参数定义为...int,表示处理一个int类型的变长参数,函数的实现,是变量所有的参数,然后累加计算平均值。 再来看看函数的调用。
Average(), 参数为0个参数
Average(1), 参数为1个参数
Average(1,6), 参数为2个参数
Average(1, 2, 3, 4, 5), 参数为5个参数
Average(n...), 参数为6个参数
注意最后一个调用参数的方式, Average(n...) n是一个数组或者是切片,可以通过...的方式,把一个数组或者切片转换为变长的参数进行传递, 这个方式在很多源代码里都有类似的调用。
在使用变长函数定义时要注意
-
一个变长参数的函数定义里,只能有一个变长的参数定义
-
一个变长参数的函数定义里,只能是最后一个参数定义能定义为变长的参数
func Says(name string, greets ...string) {
word := strings.Join(greets, " ")
fmt.Printf("%s: %s\n", name, word)
}
func TestReturnFunc2(t *testing.T) {
Says("Ginghan", "Welcome to", "Golang", "Java", "Python")
words := append(make([]string, 0), "like", "Golang", "Java", "Python", "C#", "Php", "C++", "Js")
Says("程序员紫龙", words...)
}
==== OUTPUT ====
Ginghan: Welcome to Golang Java Python
程序员紫龙: like Golang Java Python C# Php C++ Js
在上面的例子里,定义了Says变长参数的函数; 只能最后一个参数可以定义为变长的参数,如greets ...string;
words := append(make([]string, 0), "like", "Golang", "Java", "Python", "C#", "Php", "C++", "Js")
这里的代码,也是一个变长参数的函数的使用例子, golang里内置的append函数也是一个变长参数的函数, 参考builtin.go代码
// The append built-in function appends elements to the end of a slice. If
// it has sufficient capacity, the destination is resliced to accommodate the
// new elements. If it does not, a new underlying array will be allocated.
// Append returns the updated slice. It is therefore necessary to store the
// result of append, often in the variable holding the slice itself:
// slice = append(slice, elem1, elem2)
// slice = append(slice, anotherSlice...)
// As a special case, it is legal to append a string to a byte slice, like this:
// slice = append([]byte("hello "), "world"...)
func append(slice []Type, elems ...Type) []Type
这个append是处理切片非常常见的函数, 大家一定要熟记这个函数的用法。
变长的方式,只能作为变长参数,不能作为返回值。
func Says(name string, greets ...string) ...string {
word := strings.Join(greets, " ")
fmt.Printf("%s: %s\n", name, word)
}
==== Complie Error ====
.\build_test.go:546:43: can only use ... with final parameter in list
编译失败!!!