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) 
}