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主:幼麟实验室

相关内容

热门资讯

如何更换安卓系统手机,安卓系统... 你有没有想过,你的安卓手机用久了,是不是有点儿卡顿了呢?别急,今天就来教你怎么给它换换“血”,让它焕...
国家粮油统计安卓系统,智能数据... 你有没有想过,每天我们吃的粮食,背后竟然有这么多的故事和数据呢?没错,今天就要带你走进国家粮油统计的...
台电双系统安卓更新,畅享双平台... 你知道吗?最近台电的双系统安卓更新可是引起了不小的轰动呢!作为一个紧跟科技潮流的数码爱好者,我可是迫...
安卓系统上打开caj,Andr... 你有没有遇到过这种情况:手里拿着一本看起来超级有料的电子书,打开一看,哎呀妈呀,竟然是CAJ格式!别...
装安卓手机系统教程,安卓手机系... 你有没有想过,让你的安卓手机也能装上Windows系统,体验一下不一样的操作界面呢?没错,今天就要来...
安卓系统不供应华为,安卓系统不... 你知道吗?最近有个大新闻在科技圈里炸开了锅,那就是安卓系统不再供应华为了!这可不仅仅是两个公司之间的...
安卓哪个系统功耗小些,揭秘哪个... 你有没有发现,手机用着用着,电池就有点不给力了?尤其是安卓手机,系统更新换代快,但功耗问题却一直让人...
免费升级安卓系统吗,畅享智能新... 亲爱的读者们,你是否也像我一样,对安卓系统的免费升级充满了好奇和期待呢?想象你的手机从老旧的系统升级...
运行安卓app的系统,系统版本... 你有没有想过,你的手机里那些运行得风生水起的安卓应用,它们究竟是在什么样的系统下翩翩起舞的呢?今天,...
安卓系统保存照片,安卓系统照片... 手机里的照片是不是越来越多,你有没有想过,这些珍贵的回忆是怎么被安卓系统保存下来的呢?今天,就让我带...
手机安卓系统会卡吗,安卓手机系... 你有没有发现,手机用久了,尤其是安卓系统,有时候会变得有点“慢吞吞”的?这不,今天就来聊聊这个话题,...
安卓系统自己安装mim,And... 你有没有发现,你的安卓手机最近有点儿“聪明”了呢?它竟然自己动手安装了MIM(Mirroring)应...
iwork10安卓系统,创新办... 你有没有发现,最近你的手机是不是变得越来越卡了?别急,别急,让我来给你介绍一款神器——iWork10...
安卓手机装deepin系统,安... 你有没有想过给你的安卓手机换换口味呢?别再局限于那些千篇一律的操作系统了,今天就来给你揭秘如何让你的...
安卓系统不收钱吗,安卓系统免费... 你有没有想过,为什么安卓系统不用你掏腰包呢?是不是觉得它就像那免费的午餐,来者不拒,享用起来毫无负担...
8848系统是安卓的,安卓生态... 你知道吗?在手机操作系统的大千世界里,有一个特别的存在,那就是8848系统。它可不是什么小角色,而是...
安卓系统 木马删不掉,屡删不绝... 安卓系统被木马盯上的那些事儿在数字化时代,手机已经成为我们生活中不可或缺的一部分。而安卓系统,作为全...
ce系统安装安卓地图,探索智能... 最近是不是觉得手机上的地图导航有点不给力?别急,今天就来给你详细说说如何给CE系统安装安卓地图,让你...
安卓4.1系统无法升级,无法升... 你有没有遇到过这种情况?手机用了好几年,系统却怎么也升级不了?别急,今天就来聊聊这个让人头疼的安卓4...
kindle如何改安卓系统,如... 亲爱的读者们,你是否也有过这样的经历:手中的Kindle电子书阅读器突然间变得有些“不按套路出牌”,...