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

相关推荐
Wpa.wk2 小时前
自动化测试(java) - PO模式了解
java·开发语言·python·测试工具·自动化·po模式
徐先生 @_@|||2 小时前
Java/Maven 对比 Python/PyPI
开发语言·python
编程猪猪侠2 小时前
手写js轮播图效果参考
开发语言·javascript·ecmascript
思成不止于此2 小时前
C++ STL中map与set的底层实现原理深度解析
开发语言·c++·set·map·红黑树·底层实现
惺忪97982 小时前
C++ 构造函数完全指南
开发语言·c++
小此方2 小时前
Re:从零开始学C++(五)类和对象·第二篇:构造函数与析构函数
开发语言·c++
秦苒&2 小时前
【C语言】详解数据类型和变量(二):三种操作符(算数、赋值、单目)及printf
c语言·开发语言·c++·c#
无限进步_2 小时前
【C语言&数据结构】有效的括号:栈数据结构的经典应用
c语言·开发语言·数据结构·c++·git·github·visual studio
是喵斯特ya2 小时前
python开发web暴力破解工具(进阶篇 包含验证码识别和token的处理)
开发语言·python·web安全