普通函数定义
√ golang函数基本组成:关键字func、函数名、参数列表、返回值、函数体和返回语句。
• 语法如下
func 函数名(参数列表) (返回值列表) {
// 函数体
}
• 示例如下
package main import "fmt"
import "errors" func Add(a int, b int) (ret int, err error) {
if a < || b < { // 假定只支持两个非负数字的加法
err = errors.New("Should be non-negative numbers!")
return
}
return a + b, nil
} func main() {
fmt.Println(Add(-, ))
fmt.Println(Add(, ))
}
• 注意事项
▪ golang函数支持多返回值,返回时没有被明确赋值的返回值将被设置为默认值。
▪ golang函数返回值可以命名,但不是强制规则,给返回值命名可以让让代码更清晰,可读性更强,同时也可以用于文档。
▪ golang函数定义中左花括号的位置被强制规范,如果左花括号放置不规范,golang编译器报告编译错误:syntax error: unexpected semicolon or newline before {。
▪ golang函数定义中如果参数列表中若干个相邻参数的类型相同,则可以在参数列表中省略前面的变量类型声明。
func Add(a, b int) (ret int, err error) {
// 函数体
}
▪ golang函数定义中如果返回值列表中若干个相邻返回值的类型相同,则可以在返回值列表中省略前面的变量类型声明。
func Add(a, b int) (ret, err int) {
// 函数体
}
▪ golang函数定义中如果返回值只有一个,那么返回值列表可以简略表示。
func Add(a, b int) int {
// 函数体
}
固定类型的不定参数函数定义
• 语法如下
func 函数名(args ...Type) (返回值列表) {
// 函数体
}
√ 形如args ...Type的用法只能作为函数的参数类型,并且必须是最后一个参数。
√ 形如args ...Type的用法是一个语法糖(syntactic sugar),即对语言功能没有影响,但可以增加程序的可读性。
√ 形如args ...Type的用法从内部实现上说,其本质上是一个数组切片,即[]type,因此可以用for循环来获得每个传入的参数。
• 示例如下
package main import "fmt" func myfunc(prefix string, args ...int) {
fmt.Print(prefix, " : ") for _, arg := range args {
fmt.Print(arg, " ")
} fmt.Println()
} func main() {
myfunc("data1", , , , )
myfunc("data2", , , , , )
}
• 不定参数函数互相调用
▶ 原样传递
▪ 语法如下
func myfunc1(args ...Type) {
// 函数体
} func myfunc2(args ...Type) {
...
myfunc1(args...)
...
}
▪ 示例如下
package main import "fmt" func myfunc1(args ...int) {
fmt.Println("myfunc1() invoked") for _, arg := range args {
fmt.Print(arg, " ")
} fmt.Println()
} func myfunc2(args ...int) {
fmt.Println("myfunc2() invoked")
myfunc1(args...)
} func main() {
myfunc2(, , , )
}
▶ 片段传递
▪ 语法如下
func myfunc1(args ...Type) {
// 函数体
} func myfunc2(args ...Type) {
...
myfunc1(args[m:n]...)
...
}
▪ 示例如下
package main import "fmt" func myfunc1(args ...int) {
fmt.Println("myfunc1() invoked") for _, arg := range args {
fmt.Print(arg, " ")
} fmt.Println()
} func myfunc2(args ...int) {
fmt.Println("myfunc2() invoked")
myfunc1(args[:]...)
} func main() {
myfunc2(, , , )
}
任意类型的不定参数函数定义
• 语法如下
func 函数名(args ...interface{}) (返回值列表) {
// 函数体
}
√ 形如args ...interface{}的用法只能作为函数的参数类型,并且必须是最后一个参数。
• 示例如下
package main import "fmt" func MyPrintf(info string, args ...interface{}) {
fmt.Println(info)
for _, arg := range args {
switch arg.(type) {
case int:
fmt.Println(arg, "is an int value.")
case string:
fmt.Println(arg, "is a string value.")
case int64:
fmt.Println(arg, "is an int64 value.")
default:
fmt.Println(arg, "is an unknown type.")
}
}
} func main() {
var v1 int =
var v2 int64 =
var v3 string = "hello"
var v4 float32 = 1.234 MyPrintf("MyPrintf", v1, v2, v3, v4)
}
匿名函数与闭包
• 匿名函数
√ 匿名函数是指不需要定义函数名的一种函数实现方式,函数可以像普通变量一样被传递或使用。
▶ 匿名函数赋值给变量
▪ 语法如下
f := func(参数列表) (返回值列表) {
// 函数体
}
▪ 示例如下
package main import "fmt" func main() {
f := func(x, y int) int {
return x + y
} fmt.Println("Result =", f(, ))
}
▶ 匿名函数直接调用
▪ 语法如下
√ 匿名函数定义的花括号后直接跟参数列表表示函数调用。
func(参数列表) (返回值列表) {
// 函数体
} (参数列表)
▪ 示例如下
package main import "fmt" func main() {
fmt.Println("Result =",
func(x, y int) int {
return x + y
}(, ))
}
• 闭包
▶ 闭包概念
√ 闭包是可以包含*变量(未绑定到特定对象)的代码块。这些*变量不在该代码块内或任何全局上下文中定义,而是在定义代码块的环境中定义。要执行的代码块为*变量提供绑定的计算环境,即作用域。
√ *变量包含在代码块中,这些*变量以及其引用的对象将不被释放。闭包的实现确保只要闭包还被使用,那么被闭包引用的变量会一直存在。
√ golang的匿名函数就是一个闭包,闭包函数可以存储到变量中作为参数传递给其他函数,最重要的是能够被函数动态创建和返回。
√ *变量的值将被隔离,在闭包外不可修改,只有内部匿名函数才能访问,而无法通过其他途径访问,因此确保了变量的安全性。
▶ 闭包使用
▪ 语法如下
c := func() func(参数类型列表) 返回值类型列表 {
// *变量定义
...
return func(参数列表) 返回值列表 {
// 函数定义
}
}()
▪ 示例如下
package main import (
"fmt"
) func main() {
result := // 闭包定义
c := func() func(int) int {
i :=
return func(j int) int {
i +=
fmt.Printf("i = %d, j = %d\n", i, j)
return i + j
}
}() result = c()
fmt.Printf("i + j= %d\n", result) result = c()
fmt.Printf("i + j= %d\n", result) result = c()
fmt.Printf("i + j= %d\n", result)
}
main函数
√ main函数是golang可执行程序的执行起点。
√ main函数不能带有参数,命令行传入的参数在os.Args变量中保存。
package main import (
"fmt"
"os"
) func main() {
for i, v := range os.Args {
fmt.Printf("os.Args[%d] = %s\n", i, v)
}
}