Go - Go语言基础 - 函数 - 匿名函数
//squares 函数返回一个函数,后者包含下一次要用到的平方数
func main() {
f := squares()
fmt.Println(f())
fmt.Println(f())
}
func squares() func() int {
var x int
log.Println("x:",&x)
return func() int {
x++
return x*x
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
输出结果:
"1"
"4"
- 1
- 2
x变量隐藏在变量f中,并且在这个过程中只生成了一个x变量,并不是调用一次生成一个新的x变量。
Go - Go语言基础 - 函数 - 匿名函数 - 注意事项:
//匿名函数下面这种情况不可以实现递归调用
visitAll := func(items []string){
//todo
visitAll(m[item]) //compile error: undefined:visitAll
//todo
}
如果匿名函数想要实现递归调用必须采取以下写法
var visitAll func(items []string) //不可以将声明和赋值写在同一行
visitAll = func(items []string){
//todo
visitAll(m[item])
//todo
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
Go - Go语言基础 - 函数 - 捕获迭代变量
var dirs []dir
for _,dir:=range dirs{
//todo
os.RemoveAll(dir) //不正确
}
- 1
- 2
- 3
- 4
- 5
在上面的场景中不应该在循环体内直接使用dir。原因:在循环里创建的所有函数变量共享相同的变量——一个可访问的存储位置,而不是固定的值,换句话来说dir变量的值在不断地迭代更新中,因此调用清理函数的时候dir变量已经被每一次的for循环更新多次。(这种写法如果起并发可能导致错误,本人亲测)
//所以建议以下写法
var dirs []dir
for _,dir:=range dirs{
dir := dir //声明内部dir,并以外部dir初始化
//todo
os.RemoveAll(dir)
}