所谓闭包就是一个函数“捕获”了和它在同一作用域的其它常量和变量。这就意味着当闭包被调用的时候,不管在程序什么地方调用,闭包能够使用这些常量或者变量。它不关心这些捕获了的变量和常量是否已经超出了作用域,所以只有闭包还在使用它,这些变量就还会存在。

在Go语言里,所有的匿名函数(Go语言规范中称之为函数字面量)都是闭包。匿名函数是指不需要定义函数名的一种函数实现方式,它并不是一个新概念,最早可以回溯到1958年的Lisp语言。

  1. func main() {
  2. i := 0
  3. str := "mike"
  4. //方式1
  5. f1 := func() { //匿名函数,无参无返回值
  6. //引用到函数外的变量
  7. fmt.Printf("方式1:i = %d, str = %s\n", i, str)
  8. }
  9. f1() //函数调用
  10. //方式1的另一种方式
  11. type FuncType func() //声明函数类型, 无参无返回值
  12. var f2 FuncType = f1
  13. f2() //函数调用
  14. //方式2
  15. var f3 FuncType = func() {
  16. fmt.Printf("方式2:i = %d, str = %s\n", i, str)
  17. }
  18. f3() //函数调用
  19. //方式3
  20. func() { //匿名函数,无参无返回值
  21. fmt.Printf("方式3:i = %d, str = %s\n", i, str)
  22. }() //别忘了后面的(), ()的作用是,此处直接调用此匿名函数
  23. //方式4, 匿名函数,有参有返回值
  24. v := func(a, b int) (result int) {
  25. result = a + b
  26. return
  27. }(1, 1) //别忘了后面的(1, 1), (1, 1)的作用是,此处直接调用此匿名函数, 并传参
  28. fmt.Println("v = ", v)
  29. }

闭包捕获外部变量特点:

  1. func main() {
  2. i := 10
  3. str := "mike"
  4. func() {
  5. i = 100
  6. str = "go"
  7. //内部:i = 100, str = go
  8. fmt.Printf("内部:i = %d, str = %s\n", i, str)
  9. }() //别忘了后面的(), ()的作用是,此处直接调用此匿名函数
  10. //外部:i = 100, str = go
  11. fmt.Printf("外部:i = %d, str = %s\n", i, str)
  12. }

函数返回值为匿名函数:

  1. // squares返回一个匿名函数,func() int
  2. // 该匿名函数每次被调用时都会返回下一个数的平方。
  3. func squares() func() int {
  4. var x int
  5. return func() int {//匿名函数
  6. x++ //捕获外部变量
  7. return x * x
  8. }
  9. }
  10. func main() {
  11. f := squares()
  12. fmt.Println(f()) // "1"
  13. fmt.Println(f()) // "4"
  14. fmt.Println(f()) // "9"
  15. fmt.Println(f()) // "16"
  16. }

函数squares返回另一个类型为 func() int 的函数。对squares的一次调用会生成一个局部变量x并返回一个匿名函数。每次调用时匿名函数时,该函数都会先使x的值加1,再返回x的平方。第二次调用squares时,会生成第二个x变量,并返回一个新的匿名函数。新匿名函数操作的是第二个x变量。

通过这个例子,我们看到变量的生命周期不由它的作用域决定:squares返回后,变量x仍然隐式的存在于f中。

作者:admin  创建时间:2018-06-17 19:59
 更新时间:2018-06-17 20:01
上一篇:
下一篇: