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

相关推荐
疯狂的喵4 小时前
C++编译期多态实现
开发语言·c++·算法
2301_765703144 小时前
C++中的协程编程
开发语言·c++·算法
m0_748708054 小时前
实时数据压缩库
开发语言·c++·算法
感谢地心引力5 小时前
安卓、苹果手机无线投屏到Windows
android·windows·ios·智能手机·安卓·苹果·投屏
lly2024065 小时前
jQuery Mobile 表格
开发语言
惊讶的猫5 小时前
探究StringBuilder和StringBuffer的线程安全问题
java·开发语言
m0_748233176 小时前
30秒掌握C++核心精髓
开发语言·c++
Fleshy数模6 小时前
从数据获取到突破限制:Python爬虫进阶实战全攻略
java·开发语言
Duang007_6 小时前
【LeetCodeHot100 超详细Agent启发版本】字母异位词分组 (Group Anagrams)
开发语言·javascript·人工智能·python
froginwe116 小时前
Redis 管道技术
开发语言