首页 > 其他分享 >go中使用闭包

go中使用闭包

时间:2023-12-08 13:22:44浏览次数:19  
标签:闭包 函数 int func 使用 go main

Go语言中的闭包:封装数据与功能的强大工具

原创 TimLiu 爱发白日梦的后端 2023-11-01 15:35 发表于广东 收录于合集#go112个 爱发白日梦的后端 专注 Go 语言领域的发展,学习成为更牛逼的架构师,日常分享 Go 语言、架构、软件工具的使用。 149篇原创内容 公众号

闭包是包括 Go 在内的编程语言的一项强大功能。通过闭包,您可以在函数中封装数据,并通过函数的返回值访问这些数据。在本文中,我们将介绍 Go 中闭包的基础知识,包括它们是什么、如何工作以及如何有效地使用它们。

什么是闭包?

go官方有一句解释:

Function literals are closures: they may refer to variables defined in a surrounding function. Those variables are then shared between the surrounding function and the function literal, and they survive as long as they are accessible.

翻译过来就是:

函数字面量(匿名函数)是闭包:它们可以引用在周围函数中定义的变量。然后,这些变量在周围的函数和函数字面量之间共享,只要它们还可以访问,它们就会继续存在。

闭包是一种创建函数的方法,这些函数可以访问在其主体之外定义的变量。闭包是一个可以捕捉其周围环境状态的函数。这意味着函数可以访问不在其参数列表中或在其主体中定义的变量。闭包函数可以在外部函数返回后访问这些变量

在 Go 中创建闭包

在 Go 中,您可以使用匿名函数创建闭包。创建闭包时,函数会捕获其周围环境的状态,包括外部函数中定义的任何变量。闭包函数可以在外部函数返回后访问这些变量。

下面是一个在 Go 中创建闭包的示例:

func adder() func(int) int { // 外部函数
 sum := 0
 return func(x int) int { // 内部函数
  fmt.Println("func sum: ", sum)
  sum += x
  return sum
 }
}

func main() {
 a := adder()
 fmt.Println(a(1))
 fmt.Println(a(2))
 fmt.Println(a(3))
}

在本例中,我们定义了一个返回匿名函数的加法器函数。匿名函数捕捉加法器函数中定义的 sum 变量的状态。每次调用匿名函数时,它都会将参数加到求和变量中,并返回结果。

所以其输出结果为:

func sum:  0
1
func sum:  1
3
func sum:  3
6

在 Go 中使用闭包

在 Go 中,闭包可用于多种用途,包括用函数封装数据、创建生成器、迭代器和 memoization 函数。

下面是一个使用闭包将数据与函数封装在一起的示例:

func makeGreeter(greeting string) func(string) string {
 return func(name string) string {
  fmt.Printf("func greeting: %s, name: %s\n", greeting, name)
  return greeting + ", " + name
 }
}

func main() {
 englishGreeter := makeGreeter("Hello")
 spanishGreeter := makeGreeter("Hola")

 fmt.Println(englishGreeter("John"))
 fmt.Println(englishGreeter("Tim"))
 fmt.Println(spanishGreeter("Juan"))
 fmt.Println(spanishGreeter("Taylor"))
}

在本例中,我们定义了一个名为 makeGreeter 的函数,它返回一个匿名函数。该匿名函数接收一个字符串参数,并返回一个将问候语和名称连接起来的字符串。我们创建了两个问候语程序,一个用于英语,一个用于西班牙语,然后用不同的名称调用它们。

所以其输出为:

func greeting: Hello, name: John
Hello, John
func greeting: Hello, name: Tim
Hello, Tim
func greeting: Hola, name: Juan
Hola, Juan
func greeting: Hola, name: Taylor
Hola, Taylor

替换捕获的变量

Go 闭包的强大功能之一是能够更改捕获的变量。这使得代码中的行为更加灵活和动态。下面是一个例子:

func makeCounter() func() int {
 i := 0
 return func() int {
  fmt.Println("func i: ", i)
  i++
  return i
 }
}

func main() {
 counter := makeCounter()
 fmt.Println(counter())
 fmt.Println(counter())
 fmt.Println(counter())
}

在本例中,makeCounter 函数返回一个闭包,每次调用都会使计数器递增。i 变量被闭包捕获,并可被修改以更新计数器。

所以其输出为:

func i:  0
1
func i:  1
2
func i:  2
3

逃逸变量

Go 闭包的另一个高级概念是变量逃逸分析。在 Go 中,变量通常在堆栈上分配,并在超出作用域时被去分配。然而,当变量被闭包捕获时,它必须在堆上分配,以确保在函数返回后可以访问它。这会导致性能开销,因此了解变量何时以及如何逃逸非常重要。

我们对比一下两个方法:

func makeAdder1(x1 int) func(int) int {
 return func(y1 int) int {
  return x1 + y1
 }
}

func makeAdder2(x2 int) func(int) int {
 fmt.Println(x2)
 return func(y2 int) int {
  return x2 + y2
 }
}

func main() {
 a := makeAdder1(5)
 fmt.Println(a(1))

 b := makeAdder2(6)
 fmt.Println(b(1))
}

makeAdder1 和 makeAdder2 的区别在于函数内的 x 是否被使用。

而我们通过逃逸分析:

go build -gcflags "-m" main.go

会得到以下输出:

./main.go:5:6: can inline makeAdder1
./main.go:6:9: can inline makeAdder1.func1
./main.go:13:9: can inline makeAdder2.func1
./main.go:12:13: inlining call to fmt.Println
./main.go:19:17: inlining call to makeAdder1
./main.go:6:9: can inline main.makeAdder1.func1
./main.go:20:15: inlining call to main.makeAdder1.func1
./main.go:20:13: inlining call to fmt.Println
./main.go:23:13: inlining call to fmt.Println
./main.go:6:9: func literal escapes to heap
./main.go:12:13: ... argument does not escape
./main.go:12:14: x2 escapes to heap
./main.go:13:9: func literal escapes to heap
./main.go:19:17: func literal does not escape
./main.go:20:13: ... argument does not escape
./main.go:20:15: ~R0 escapes to heap
./main.go:23:13: ... argument does not escape
./main.go:23:15: b(1) escapes to heap

从逃逸分析结果来看,x 变量被闭包捕获,必须在堆上分配。不过,如果 x 变量不被闭包之外的任何其他代码使用,编译器可以进行优化,将其分配到栈中。

共享闭包

最后,Go 中的闭包可以在多个函数之间共享,从而实现更高的灵活性和模块化代码。下面是一个例子:

type Calculator struct {
 add func(int, int) int
}

func NewCalculator() *Calculator {
 c := &Calculator{}
 c.add = func(x, y int) int {
  fmt.Printf("func x: %d, y: %d\n", x, y)
  return x + y
 }
 return c
}

func (c *Calculator) Add(x, y int) int {
 return c.add(x, y)
}

func main() {
 calc := NewCalculator()
 fmt.Println(calc.Add(1, 2))
 fmt.Println(calc.Add(2, 3))
}

在本例中,Calculator 结构具有一个 add 函数,该函数在 NewCalculator 函数中通过闭包进行了初始化。Calculator 结构的 Add 方法只需调用 add 函数,这样就可以在多个上下文中重复使用。

所以其输出为:

func x: 1, y: 2
3
func x: 2, y: 3
5

结论

在 Go 编程中,闭包是一个强大的工具,可用于用函数封装数据,并创建生成器和迭代器等。它们提供了一种访问函数体外定义的变量的方法,即使在函数返回后也是如此。

关注和添加微信

进技术交流群,带你徜徉在知识的海洋中

图片

 

 

TimLiu

赞赏二维码喜欢作者

阅读 318 爱发白日梦的后端 ​ 喜欢此内容的人还喜欢   「Go开源」goose:深入学习数据库版本管理工具     我看过的号 爱发白日梦的后端 不看的原因   云原生系列Go语言篇-泛型     AlanHou 不看的原因   8个流行的Python可视化工具包     潮汕IT智库 不看的原因   关注公众号后可以给作者发消息          

人划线

标签:闭包,函数,int,func,使用,go,main
From: https://www.cnblogs.com/cheyunhua/p/17885936.html

相关文章

  • 代码漏洞扫描工具sonarqube在本地环境的使用
    sonarqube可以与源码管理工具gitlab集成,实现提交代码后自动扫描检测代码的相关漏洞。该CI/CD过程大致为:1、研发人员提交源码至gitlab服务器—>2、gitlabrunner执行指定脚本(由项目的.gitlab-ci.yml配置文件指定具体内容,如编译项目、开启代码检测) —>3、sonar-scanner对项目进......
  • HTML中title标签的使用
    HTML中的title标签是非常重要的标签之一,它用来描述网页的标题。在搜索引擎优化中,title标签是非常关键的,因为搜索引擎会将title标签中的文字作为页面的主要描述,并根据其相关性来判断网页内容的质量和权重。本文将详细讲解title标签的使用,包括以下内容:@[toc]##1.title标签的基本用......
  • centos安装xrdp服务,可以使用系统用户mstsc连接
    Centos6安装依赖yuminstall-yautoconfautomakelibtoolpkg-configopenssl-develpam-devellibjpeg-develfuse-develTurboJPEGlibX11-devellibXfixes-devellibXrandr-develnasmbinutilsredhat-lsb-coreCentos7安装依赖yuminstall-yfingercmakepatchgccma......
  • Navicat 使用笔记
    自动备份(自动运行)Navicat有备份功能,但要达到自动备份如每日00:30备份数据库,就要用到自动运行功能。首先要设置备份保存路径,仅能对数据库服务器进行设置而不能对具体的一个数据库设置,设置方法是右键点击数据库连接,选择编辑连接,找到高级-设置位置,修改或保持默认。点击工具栏自......
  • Windows服务器,通过Nginx部署VUE+Django前后端分离项目
    目录基本说明安装Nginx部署VUE前端部署Django后端Djangoadmin静态文件(CSS,JS等)丢失的问题1.基本说明本文介绍了在windows服务器下,通过Nginx部署VUE+Django前后端分离项目。本项目前端运行在80端口,服务器端运行在8000端口。因此本项目使用Django的......
  • go-zero 开发入门-API服务开发示例
    接口定义定义API接口文件接口文件add.api的内容如下:syntax="v1"info(title:"API接口文件示例"desc:"演示如何编写API接口文件"author:"一见"date:"2023年12月07日"version:"v1")typeAddReq......
  • 【机器学习】Django,余弦距离之基于用户,评分物品的推荐
    表设计#用户表classUserInfo(models.Model):username=models.CharField(max_length=32,unique=True,verbose_name="用户名")password=models.CharField(max_length=64)#物品表classMovies(models.Model):name=models.CharField(max_length=255,v......
  • 使用 Kubernetes 为 CI/CD 流水线打造高效可靠的临时环境
    介绍在不断发展的科技世界中,快速构建高质量的软件至关重要。在真实环境中测试应用程序是及早发现和修复错误的关键。但是,在真实环境中设置CI/CD流水线进行测试可能既棘手又昂贵。 Kubernetes是一个流行的容器编排平台,提供临时环境解决方案。在Kubernete的帮助下,用户能根......
  • xcat docker部署使用
    已打包到docker镜像dockerpullleaus/xcat:2.14.6#softversion:2.14.6#osversion:centos7.6.1610本镜像仅适用于centos7以上系统,支持docker共用宿主机网络宿主机不能存在tftpd、dhcpd、httpd、chronyd服务,否则可能会导致xcat启动失败宿主机免密码登录(已有可跳过)ssh......
  • vue中this.$refs的使用方法和遇到的问题
    this.$refs:用于操作真实的DOM节点。 在开发时碰到了一个小需求,需要子组件向父组件传参,而且是不需要通过事件传递的,一开始使用this.$emit()来写的,但是一直没有接受到参数,于是放弃了使用this.$emit()的使用。 于是,使用了在父组件中调用子组件的方法,来获取传递的参数。 一.......