Go语言之闭包
创始人
2024-06-02 17:22:51
0

一、闭包定义

1.简单的闭包场景

  • 简单来说,一个函数采用了外部的局部变量,就叫闭包
  • 如下面,main函数中匿名函数使用了函数外部定义的局部变量ret,该匿名函数就被称为闭包函数,该变量ret被称为捕获变量
package mainimport "fmt"func main(){ret := 0res := func(int)int{ret = ret + 3fmt.Println(ret)return ret}fmt.Println(res(1))
}

2.复杂的闭包场景

package mainimport "fmt"func cfunc()func()int{i := 4res := func()int{return i}return res
}
func main(){fres1 := cfunc()fres2 := cfunc()fmt.Println(fres1())fmt.Println(fres2())
}
  • 上述代码结果应该输出什么呢?结果是fres1和fres2都是5,这个是为什么呢?这不得不先说下闭包的原理了。

二、闭包原理

1.捕获变量除初始化赋值外没有被修改

  • 我们就以上述复杂场景来简单说下闭包的原理
    在这里插入图片描述
  • 上述代码在执行阶段,调用cfunc函数就会在堆上创建一个闭包函数的funcval结构体(该结构体中存有fn是闭包函数的地址和捕获列表)。这个结构体的起始地址作为返回值写入返回值空间。通过fres1、fres2调用闭包函数,就会找到各自的funcval结构体,拿到同一个函数入口。因为通过fres1、fres2找到的都是各自的funcval结构体,故拿的都是各自的捕获列表。这就是称闭包函数为有状态的函数的原因。
  • 闭包函数是如何找到对应的捕获列表?在go中通过funcval结构体调用函数时,会把funcval该结构体的地址放到一个寄存器中。这样在闭包函数中就可以通过寄存器取出funcval结构体的地址,然后加上对应的偏移量找到捕获列表里每个捕获变量。
  • 捕获列表并不是拷贝变量的值这么简单。被闭包捕获的变量要在外层函数与闭包函数中表现一致,就好像外层函数和闭包函数使用的是同一个变量一样。对此Go语言编译器做了不同的几种处理:
    (1)被捕获的变量除了初始化赋值外,在任何地方都没有被修改过,这种情况,直接是将值拷贝到捕获列表中即可
    (2)若除了初始化赋值外还被修改过,则细分为三种情况:变量逃逸、参数堆分配、返回值

2.变量逃逸

在这里插入图片描述

  • 上述例子中,被捕获的局部变量i除了初始化赋值后还被修改,此时在funcval闭包对象结构体中,捕获列表存的是捕获变量的地址(&i)。
  • 定义fres阶段:cfunc栈帧中,由于变量被捕获,局部变量i会改为堆分配,在栈上只存i的地址(&i)。第一次for循环,在堆上创建funcval结构体,捕获i的地址,这样闭包函数和外层函数就操作同一个变量了,返回值第一个元素存储addr0,第一次for循环执行结束,i++从0变成1。第二次for循环开始,在堆上创建funcval结构体,捕获i的地址,返回值第二个元素存储addr1,第二次for循环执行结束,i++从1变成2。第三次for循环开始,在堆上创建funcval结构体,捕获i的地址,返回值第二个元素存储addr2,第三次for循环执行结束,i++从2变成3。满足for循环退出条件,cfunc函数执行结束,把返回值拷贝到局部变量fres。
  • 通过fres去调用cfunc函数阶段:通过fres[0]调用cfunc函数时,把闭包结构体的起始地址addr0存入寄存器中,闭包函数通过寄存器存储的地址加上偏移找到捕获变量i的地址;通过fres[1]调用cfunc函数时,把闭包结构体的起始地址addr1存入寄存器中,闭包函数通过寄存器存储的地址加上偏移找到捕获变量i的地址;通过fres[2]调用cfunc函数时,把闭包结构体的起始地址addr2存入寄存器中,闭包函数通过寄存器存储的地址加上偏移找到捕获变量i的地址;被捕获的i的地址都是同一个,故打印的都是3。

3.参数堆分配

在这里插入图片描述

  • 若修改和被捕获的是参数,涉及到函数原型,参数依然通过调用者栈帧传入,但是编译器会把栈上的参数拷贝到堆上,然后外层函数和闭包函数都使用堆上分配的这一个。

4.返回值

在这里插入图片描述

  • 若处理的是返回值,调用者栈帧上依然会分配返回值空间,不过闭包的外层函数会在堆上也分配一个返回值空间。外层函数和闭包函数都使用堆上这个,但在外层函数返回前,需要把堆上的这个返回值拷贝到栈上的返回值空间中

参考:b站up主:幼麟实验室

相关内容

热门资讯

瑞天 图书 收钱吗-瑞天书店,... 哎呀,说到这个瑞天书店,我真是哭笑不得!你知道吗,每次我走进去,心里都在打鼓,这家店到底收不收钱?首...
朝阳医院电话是多少-朝阳医院电... 哎呀,说到朝阳医院的电话,我可是有一肚子的话要说!你知道吗,每次有个头疼脑热,我就得翻箱倒柜找那张写...
触动内心的悸动画:简单风景与故... 在这个喧嚣的世界里,总有一些画面能够触动内心最柔软的地方,它们就像是一幅幅悸动画,让我们的心跳不由自...
gps精确度-GPS 导航:是... 哎呀,说到GPS,我这心里就五味杂陈啊!你有没有过那种经历,明明跟着GPS走,结果却走进了个死胡同,...
safari浏览器书签同步-S... 嘿,亲爱的Safari用户们,你们有没有经历过那种心碎的瞬间——当你在一台新电脑上打开Safari,...
c反应蛋白肺炎-C 反应蛋白:... 哎呀,最近这C反应蛋白啊,简直成了我们健康的晴雨表。一听到这个名字,心里就不由得咯噔一下。这不,最近...
recover4all怎么注册... 哎呀,说到这个Recover4All注册,真是让人又爱又恨!你想啊,这个软件这么神奇,能帮你找回那些...
ubuntu 17.04 dv... 哟,大家好!今天我要聊聊我那个闪闪发光的宝贝——Ubuntu17.04DVD。这可不是普通的DVD哦...
刺客信条以停止工作-刺客信条总... 哎呀,我的天!刺客信条,你这是怎么了?每次我兴冲冲地点开你,想要在那些古老的街巷里飞檐走壁,你却总是...
盗版gta5mod安装教程-盗... 哎呀,说到这个盗版GTA5Mod安装教程,我真的气不打一处来!你们这些新手玩家,听我说,千万别被那些...
警惕!可移动磁盘病毒来袭,如何... 哎呀,说到这个可移动磁盘病毒,我就一肚子火!你想想,本来这小小的U盘,是我们传递文件的好帮手,现在倒...
监控布线图解-探秘监控布线世界... 大家好,我是你们的“探秘小能手”,今天我们要一起潜入那些神秘的监控布线世界,揭开它们的神秘面纱!是不...
洋地黄类药物治疗护理-洋地黄:... 哎呀,说到洋地黄,这可是我们医护人员心中的“老朋友”了!别看它名字土土的,作用可大着呢!这小家伙,专...
逆魂界7.25攻略-7.25 ... 嘿,各位逆魂界的战士们!今天我要带你们深入7.25版本的攻略,这可不是一般的战斗,这是热血与汗水的较...
克罗恩病是重大疾病吗-克罗恩病... 克罗恩病,这个名字听起来就像是来自某个遥远星球的神秘疾病,但实际上,它就在我们身边,悄无声息地侵蚀着...
电脑换64位系统怎么换-电脑换... 哎呀,说到电脑换64位系统,我这心里就像过山车一样,七上八下的。刚开始,我还以为这事儿多难呢,结果一...
qq空间 邮箱无法打开-QQ ... 哎哟喂,今天真是倒霉透顶了!我刚刚想查一下我的QQ空间邮箱,看看有没有什么重要消息,结果呢?页面愣是...
数组初始化为1-编程世界里数组... 哎呀,说到数组初始化为1,我这心里就有点儿小情绪!你说说,这编程世界里,数组咋就这么爱1呢?每次新建...
aceru盘重装系统-Acer... 大家好,我今天真的是要哭出来了!我的Acer笔记本最近卡得不行,打开个网页都要等半天,简直让人抓狂。...
赵丽颖雅虎搜星视频-重温雅虎搜... 哎呀,说到赵丽颖,我就忍不住激动起来!你们看过她在雅虎搜星的视频吗?那简直是太迷人了!记得第一次看到...