匿名函数和闭包
一、匿名函数
- 区别于普通的具名函数,匿名函数无需声明函数名,带函数体即可
- 匿名函数的优势就在于,不用申明一个函数,即可使用函数内的变量
- 当匿名函数引用了外部作用域中的变量时就成了闭包函数
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| package main
import ( "fmt" "math" )
func main() { getSqrt := func(a float64) float64 { return math.Sqrt(a) } fmt.Println(getSqrt(4)) }
|
二、闭包
什么是闭包?
- 闭包是由函数及其相关引用环境组合而成的实体
- 当匿名函数引用了外部作用域中的变量时就成了闭包函数
- 个人感觉,从模块和作用域的角度看,闭包和类有点类似,包含变量和方法,方法可以引用类里面的变量;
- 闭包的目的就是减少全局变量的使用,但是不够明确,一般不推荐用
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
| package main
import ( "fmt" )
func a() func() int { i := 0 fmt.Printf("i(%p)=%d\n",&i, i ) b := func() int { fmt.Printf("i(%p)=%d\n",&i, i ) i++ fmt.Println(i) return i } return b }
func main() { c := a() c() c() c() a() a()() }
|
闭包复制的是原对象的指针
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| package main
import "fmt"
func test() func() { x := 100 fmt.Printf("x (%p) = %d\n", &x, x)
return func() { fmt.Printf("x (%p) = %d\n", &x, x) } }
func main() { f := test() f() }
x (0xc42007c008) = 100 x (0xc42007c008) = 100
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| func add(base int) func(int) int { return func(i int) int { base += i return base } } func main() { t1 := add(10) fmt.Println(t1(1), t1(2))
t2 := add(100) fmt.Println(t2(1), t2(2)) }
|
Q: 闭包函数是函数式编程语言的核心?
三、迭代中的变量
由于循环变量的作用域的规则限制,在循环里创建的所有函数变量共享相同的变量 (一个可访问的存储位置,而不是固定的值);所有函数取到的值是最后一次循环的值
3.1.Example
现在想创建三个临时目录,然后用完之后将这三个临时目录都删除。
按下面这么写实际上只会删除最后一个目录 tmp3
:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| package main
import "os"
func main() { var rmDirs []func() tempDirs := []string {"tmp1", "tmp2", "tmp3"} for _, dir := range tempDirs{ os.MkdirAll(dir, 0755) rmDirs = append(rmDirs, func(){ os.RemoveAll(dir) }) }
for _, rmdir := range rmDirs { rmdir() } }
|
1 2 3 4 5
| tree . ├── main.go ├── tmp1 └── tmp2
|
需要设置一个中间变量,才能实现效果
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| package main
import "os"
func main() { var rmDirs []func() tempDirs := []string {"tmp1", "tmp2", "tmp3"} for _, d := range tempDirs{ dir := d os.MkdirAll(dir, 0755) rmDirs = append(rmDirs, func(){ os.RemoveAll(dir) }) }
for _, rmdir := range rmDirs { rmdir() } }
|