当前位置: 首页 > news >正文

【橘子golang】从golang来谈闭包

一、简介

闭包(Closure)是一种编程概念,它允许函数捕获并记住其创建时的上下文环境(包括变量)。闭包通常用于函数式编程语言,但在许多现代编程语言中也有支持,包括 Go ,Js等支持函数式编程的编程语言。
说的简单一点就是,闭包是一个函数值,它引用了其创建时的自由变量(即非局部变量)。当闭包被调用时,它会访问这些变量的值,即使这些变量在闭包创建时的作用域已经结束。闭包的核心特性包括:
函数值:闭包本质上是一个函数。
捕获上下文:闭包可以捕获并记住其创建时的变量状态。
持久化变量:即使创建闭包的作用域已经结束,闭包仍然可以访问这些变量。
是不是看的很头大,我们来用代码来描述一下。

二、golang中的闭包

golang作为一个现代编程语言,虽然众多褒贬不一但是他确实支持了很多现代编程语言的特点,比如函数编程。

package main

import (
	"fmt"
)

// 定义一个函数 addCounter,它返回一个匿名函数(闭包)。
func addCounter() func() {
	// 定义一个局部变量 count,它是一个整数,初始值为 0。
	// 这个变量是闭包的“状态”,它会被闭包捕获。
	var count int = 0

	// 返回一个匿名函数,这个匿名函数可以访问并修改外部的 count 变量。
	return func() {
		// 每次调用这个匿名函数时,count 的值会加 1。
		count += 1
		// 打印当前 count 的值。
		fmt.Printf("count is %d\n", count)
	}
}

func main() {
	var add1 = addCounter()
	add1() // 输出: 1
	add1() // 输出: 2
}

这个函数addCounter的特点如下:
1、名称为addCounter
2、入参为空,该函数没有入参。
3、返回值是一个函数,这个返回值函数的类型就更简单了,入参和出参都没有。

但是重点就在于这个addCounter的返回函数,我们来单独看这一段逻辑(我们称之为内部函数):

return func() {
	// 闭包函数
	count += 1
	fmt.Printf("count is %d\n", count)
}

逻辑很简单,就是对count变量做了一个+1的操作,但是这个count不是这个返回函数里面,他是在他的外部函数addCounter中声明的,而这个内部函数就是闭包,返回一个匿名函数(闭包)。这个匿名函数可以访问并修改 addCounter 中的 count 变量。该闭包捕获了来自外部函数的变量,当捕获完成后,此时该变量的生命周期就发生了变化。
具体我们再往下看:
我们在主函数中调用逻辑:

func main() {
	var add1 = addCounter()
	add1() // 输出: 1
	add1() // 输出: 2
}

我们调用var add1 = addCounter(),返回了一个add1变量,此时这就调用了这个外部函数,并且返回了内部函数给add1,注意,此时addCounter()函数调用已经结束了,那么在我们常规的理解中,当一个函数调用结束,函数中创建的变量自然也是结束了,被销毁了(我们没有返回)。但是闭包这里不同,闭包在创建的时候捕获了这个变量count,此时虽然 addCounter调用结束了,但它捕获了 addCounter 中的 count 变量。即使 addCounter 函数的执行结束,count 变量仍然被匿名函数记住,并且可以在后续调用中继续使用。这就是闭包的核心,闭包捕获外部变量,即便外部变量所在的函数已经调用结束了,但是闭包依然可以使用该变量。这就是我们为啥看到的输出1,2.
因为当var add1 = addCounter() 执行,第一个闭包被创建,也就是add1,我们多次执行该闭包,都是能访问该被闭包捕获的变量的。

但是如果我们改成这样。

func main() {
	var add1 = addCounter()
	add1() // 输出: 1
	var add2 = addCounter()
	add2() // 输出: 1
}

此时就会输出两个1,因为你第二次调用var add2 = addCounter(),此时创建的是一个新的闭包,并将返回的闭包赋值给变量 add2。这里的 add2 是一个独立的闭包,它有自己的 count 变量,重新初始化为0,所以就输出都是1。

这基本就是闭包的概念,这个概念也是因为内部匿名函数封闭并包围作用域中的变量而得名的。

三、总结

闭包的主要特点就是:

捕获变量:闭包捕获了其创建时的变量(如 count),即使这些变量在其原始作用域中已经不可访问,就像我们初始化count的函数addCounter已经结束了,但是他还是被闭包捕获到了。

独立状态:每个闭包实例都有自己的独立状态。add1 和 add2 是两个独立的闭包,它们各自维护自己的 count 变量。

延迟求值:闭包在被调用时才计算其捕获的变量的值,而不是在创建时。函数式编程的特点大多如此。我们在java中使用lambda作为参数的时候也是如此。

go语言编程规范
effective_go

相关文章:

  • 【五.LangChain技术与应用】【29.LangChain Agent小案例1:智能代理的实战应用】
  • 6. 机器人实现远程遥控(具身智能机器人套件)
  • 【JAVA架构师成长之路】【持久层】第5集:PreparedStatement防SQL注入
  • 计算机数据库三级刷题总结(博主89分已过,总结的内容分享)
  • Linux 开发工具
  • 学习使用ESP8266进行MQTT通信并在网页上可视化显示
  • VEC系列-RabbitMQ 入门笔记
  • 三款好用远程终端软件
  • FPGA时序约束的几种方法
  • MAVEN手动配置(阿里云)全教程
  • 【Linux———信号精讲】
  • DELETE/ UPDATE/ INSERT 语句会自动加锁
  • HarmonyOS:基于hmrouter实现Page的生命周期监听
  • 【监督学习】支持向量机步骤及matlab实现
  • Ruby爬虫如何控制并发数量:爬取京东电子产品
  • 在 Docker 中,无法直接将外部多个端口映射到容器内部的同一个端口
  • 前端多角色权限页面(同浏览器同时登录)数据互串解决
  • 常见的限流算法有哪些
  • Redis 面试专题
  • vue2.6附件预览及下载
  • 网络科技公司名称大全/搜索引擎优化概述
  • java做网站seo/培训心得体会总结
  • 蚌埠百度做网站/给公司建网站需要多少钱
  • 网站首页包括哪些内容/郑州网站策划
  • 网站建设官方网站/企业推广软件
  • 泰安做网站/如何优化关键词