|
普通函数定义
√ golang函数基本组成:关键字func、函数名、参数列表、返回值、函数体和返回语句。
· 语法如下
func 函数名(参数列表) (返回值列表) {
// 函数体
}
· 示例如下
package main
import
"fmt"
import
"errors"
func
Add(a int, b int) (ret int, err error) {
if a < 0 || b < 0 { // 假定只支持两个非负数字的加法
err = errors.New("Should be non-negative numbers!")
return
}
return a + b, nil
}
func main() {
fmt.Println(Add(-1, 1))
fmt.Println(Add(10, 1))
}
· 注意事项
▪ 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", 1, 2, 3, 4)
myfunc(
"data2", 4, 3, 2, 1, 0)
}
· 不定参数函数互相调用
▶ 原样传递
▪ 语法如下
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(
1, 2, 3, 4)
}
▶ 片段传递
▪ 语法如下
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[
2:3]...)
}
func main() {
myfunc2(
1, 2, 3, 4)
}
任意类型的不定参数函数定义
· 语法如下
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 = 1
var v2 int64 = 234
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(1, 2))
}
▶ 匿名函数直接调用
▪ 语法如下
√ 匿名函数定义的花括号后直接跟参数列表表示函数调用。
func(参数列表) (返回值列表) {
// 函数体
} (参数列表)
▪ 示例如下
package main
import
"fmt"
func
main() {
fmt.Println(
"Result =",
func(x, y
int) int {
return x + y
}(
1, 2))
}
· 闭包
▶ 闭包概念
√ 闭包是可以包含自由变量(未绑定到特定对象)的代码块。这些自由变量不在该代码块内或任何全局上下文中定义,而是在定义代码块的环境中定义。要执行的代码块为自由变量提供绑定的计算环境,即作用域。
√ 自由变量包含在代码块中,这些自由变量以及其引用的对象将不被释放。闭包的实现确保只要闭包还被使用,那么被闭包引用的变量会一直存在。
√ golang的匿名函数就是一个闭包,闭包函数可以存储到变量中作为参数传递给其他函数,最重要的是能够被函数动态创建和返回。
√ 自由变量的值将被隔离,在闭包外不可修改,只有内部匿名函数才能访问,而无法通过其他途径访问,因此确保了变量的安全性。
▶ 闭包使用
▪ 语法如下
c :=func() func(参数类型列表) 返回值类型列表 {
// 自由变量定义 ...
returnfunc(参数列表) 返回值列表 {
// 函数定义
}
}()
▪ 示例如下
package main
import (
"fmt"
)
func main() {
result :
= 0
// 闭包定义
c := func() func(int) int {
i := 0
return func(j int) int {
i += 10
fmt.Printf("i = %d, j = %d\n", i, j)
return i + j
}
}()
result = c(10)
fmt.Printf("i + j= %d\n", result)
result = c(20)
fmt.Printf("i + j= %d\n", result)
result = c(30)
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)
}
}
|
|