序言
在GoLang开发过程中,经常会遇到「匿名函数」和「闭包」,但是对这两个概念始终是一知半解,因此决定认真整理一下以解答这一疑惑。
一、什么是匿名函数?
首先,什么是「匿名函数」。匿名函数,顾名思义就是没有名字的函数,我们从一个简单的例子开始介绍。
1. 不带参数的匿名函数
package main
import (
"fmt"
)
func main() {
f:=func(){
fmt.Println("hello world")
}
f()//hello world
fmt.Printf("%T\n", f) //打印 func()
}
在该例子中,f 的值即为一个匿名函数,要调用该函数直接使用 f() 即可,此时 f 成为了事实上的函数名(注:%T是用来输出 Go 语言语法格式的类型和值)。
2. 带参数的匿名函数
接下来我们给上面这个匿名函数加上参数。
// 带参数的匿名函数
package main
import (
"fmt"
)
func main() {
f:=func(args string){
fmt.Println(args)
}
f("hello world")//hello world
//或
(func(args string){
fmt.Println(args)
})("hello world")//hello world
//或
func(args string) {
fmt.Println(args)
}("hello world") //hello world
}
3. 带返回值的匿名函数
然后再是加上返回值的匿名函数。
// 带返回值的匿名函数
package main
import "fmt"
func main() {
f:=func()string{
return "hello world"
}
a:=f()
fmt.Println(a)//hello world
}
当程序需要完成某一部分逻辑同时逻辑不是太复杂时,可以直接嵌入一个匿名函数,省去独立命名函数的麻烦。并且由于匿名函数类似于C++中的inline内联,速度表现会更快。但是需要注意的是代码在80%的时间是被阅读,因此若逻辑较复杂则不要使用匿名函数。
什么是闭包?
「闭包」,即函数的嵌套。内层的函数可以使用外层函数的所有变量,即使外层函数已经执行完毕。还是从一个简单的例子开始介绍。
func incr() func() int {
var x int
return func() int {
x++
return x
}
}
调用这个函数会返回一个函数变量。
i := incr():通过把这个函数变量赋值给 i,i 就成为了一个闭包。
所以 i 保存着对 x 的引用,可以想象 i 中有着一个指针指向 x 或 i 中有 x 的地址。
由于 i 有着指向 x 的指针,所以可以修改 x,且保持着状态:
println(i()) // 1
println(i()) // 2
println(i()) // 3
因此一句话总结匿名函数和闭包的关系,就是匿名函数是闭包的组成部分。
参考:
[1] https://juejin.cn/post/6844903793771937805
[2] https://segmentfault.com/a/1190000018689134