Go基础-闭包

什么是闭包?

闭包的应该都听过,但到底什么是闭包呢?

闭包是由函数及其相关引用环境组合而成的实体(即:闭包=函数+引用环境)。

"官方"的解释是:所谓"闭包",指的是一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分。

维基百科讲,闭包(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里面怎么实现怎么调用。本质上说,在匿名函数里使用了外部的变量,才叫闭包

相关推荐
阿巴斯甜9 小时前
Android 报错:Zip file '/Users/lyy/develop/repoAndroidLapp/l-app-android-ble/app/bu
android
Kapaseker10 小时前
实战 Compose 中的 IntrinsicSize
android·kotlin
xq952711 小时前
Andorid Google 登录接入文档
android
黄林晴12 小时前
告别 Modifier 地狱,Compose 样式系统要变天了
android·android jetpack
冬奇Lab1 天前
Android触摸事件分发、手势识别与输入优化实战
android·源码阅读
城东米粉儿1 天前
Android MediaPlayer 笔记
android
Jony_1 天前
Android 启动优化方案
android
阿巴斯甜1 天前
Android studio 报错:Cause: error=86, Bad CPU type in executable
android
张小潇1 天前
AOSP15 Input专题InputReader源码分析
android
_小马快跑_1 天前
Kotlin | 协程调度器选择:何时用CoroutineScope配置,何时用launch指定?
android