当通过调用外部函数返回的内部函数后,即使外部函数已经执行结束了,但是被内部函数引用的外部函数的变量依然会保存在内存中,我们把引用了其他函数作用域变量的函数和这些被引用变量的集合,称为闭包(Closure),闭包是这些东西共同的组合
在了解闭包的概念和用途之前,理解作用域和变量的生命周期等基础预备知识,对于理解闭包非常有帮助。
-
闭包的三大特性
⭐ 函数嵌套函数
⭐ 函数内部可以引用外部的参数及变量
⭐ 参数和变量不会被垃圾回收机制回收
-
实现
Go Language Infrastructure(GLI)1
-
优点
🔺 可创建私有变量
通过使用闭包,可以在外部调用闭包函数,从而在函数外部能够间接访问到函数内部的变量,也可以使用这种方法来创建私有变量避免全局变量的污染。
🔺 让这些变量的值始终保持在内存中
使已经运行结束的函数上下文中的变量对象继续留在内存中,因为闭包函数保留了这个变量对象的引用,所以这个变量对象不会被垃圾回收机制回收实现变量数据共享。
-
缺点
- 滥用闭包会导致大量变量不会被垃圾回收机制回收,内存消耗很大,可能会造成网页卡顿等性能问题
- 函数内部的局部变量没有被释放,使得占用内存时间会变长,容易造成内存泄漏。
- 闭包会在父函数外部,改变父函数内部变量的值。若将函数当作 Object 使用,把闭包当作它的共用方法,内部变量当作它的私有属性,这时一定要小心,不要随便改变父函数内部变量的值。
Go Language Infrastructure(GLI)
5. Closures(闭包)
Go 支持匿名函数,这些函数可以形成闭包。当您希望在不命名的情况下内联定义函数时,匿名函数非常有用。
这个函数 intSeq 返回另一个函数,我们在 intSeq 的主体中匿名定义它。返回的函数关闭变量 i 以形成一个闭包。
我们调用 intSeq,将结果(一个函数)分配给 nextInt。这个函数值捕获它自己的 i 值,每次调用 nextInt 时都会更新它。
通过调用 nextInt 几次来查看闭包的效果。
package main import "fmt" func intSeq() func() int { i := 0 return func() int { i++ return i } } func main() { nextInt := intSeq() fmt.Println(nextInt()) fmt.Println(nextInt()) fmt.Println(nextInt()) newInts := intSeq() fmt.Println(newInts()) }
output:
↩$ go run closures.go 1 2 3 1
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于