什么是闭包?
闭包的应该都听过,但到底什么是闭包呢?
闭包是由函数及其相关引用环境组合而成的实体(即:闭包=函数+引用环境)。
"官方"的解释是:所谓"闭包",指的是一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分。
维基百科讲,闭包(Closure),是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。闭包在运行时可以有多个实例,不同的引用环境和相同的函数组合可以产生不同的实例。
看着上面的描述,会发现闭包和匿名函数似乎有些像。可是可能还是有些云里雾里的。因为跳过闭包的创建过程直接理解闭包的定义是非常困难的。目前在JavaScript、Go、PHP、Scala、Scheme、Common Lisp、Smalltalk、Groovy、Ruby、 Python、Lua、objective c、Swift 以及Java8以上等语言中都能找到对闭包不同程度的支持。通过支持闭包的语法可以发现一个特点,他们都有垃圾回收(GC)机制。 javascript应该是普及度比较高的编程语言了,通过这个来举例应该好理解写。看下面的代码,只要关注script里方法的定义和调用就可以了。
闭包如何实现
我们实现一个延迟计算总和的函数
go
package main
import (
"fmt"
"time"
)
func main() {
//add := awaitAdd(2)
t1 := time.Now()
sum := awaitAdd(2)(1, 2, 3)
//sum := add(1, 2, 3)
since := time.Since(t1)
fmt.Println(sum, since)
}
func awaitAdd(await int) func(...int) int {
// 延时函数放在上面和放在下面是有区别的
fmt.Println("先执行await add")
time.Sleep(time.Duration(await) * time.Second)
return func(num ...int) int {
//time.Sleep(time.Duration(await) * time.Second)
var sum int
for _, v := range num {
sum += v
}
return sum
}
}
从上面的代码可以看出,闭包就是是在函数里面套一个函数,并且再把这个函数作为返回值返回。但是闭包可以被传递给其他函数,作为函数调用的参数;闭包可以被函数返回,作为函数的返回值。关键是作为外部函数,它可以引用函数参数局部变量。
小问
这两个延迟有没有区别?

第一种测试
在匿名函数里延迟
go
func awaitAdd(await int) func(...int) int {
// 延时函数放在上面和放在下面是有区别的
fmt.Println("先执行await add")
//time.Sleep(time.Duration(await) * time.Second)
return func(num ...int) int {
time.Sleep(time.Duration(await) * time.Second)
var sum int
for _, v := range num {
sum += v
}
return sum
}
}
主函数调用
go
func main() {
add := awaitAdd(2)
t1 := time.Now()
//sum := awaitAdd(2)(1, 2, 3)
sum := add(1, 2, 3)
since := time.Since(t1)
fmt.Println(sum, since)
}
打印结果:
powershell
先执行await add
6 2.0005028s
第二种测试
go
func awaitAdd(await int) func(...int) int {
// 延时函数放在上面和放在下面是有区别的
fmt.Println("先执行await add")
time.Sleep(time.Duration(await) * time.Second)
return func(num ...int) int {
//time.Sleep(time.Duration(await) * time.Second)
var sum int
for _, v := range num {
sum += v
}
return sum
}
}
主函数不变,看打印结果:
powershell
先执行await add
6 0s
总结
关于闭包我一直都是一知半解,哪怕在此刻也还是有点模糊。但是从这里能看出来,闭包的本身就是函数调内部函数,我这里记录的就是闭包在go里面怎么实现怎么调用。本质上说,在匿名函数里使用了外部的变量,才叫闭包