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

相关内容

热门资讯

安卓挖煤模式重置系统,系统重置... 手机突然卡壳了,系统也变得不认路了,你是不是也遇到了这样的烦恼?别急,今天就来给你揭秘安卓手机的“挖...
安卓系统8和5,跨越时代的系统... 你有没有发现,手机里的那个安卓系统,就像是我们的好朋友,总在默默无闻地陪伴着我们。今天,咱们就来聊聊...
安卓系统程序自启动,安卓系统程... 你有没有发现,你的安卓手机有时候就像一个超级忙碌的小蜜蜂,不管你开不开它,它总是自顾自地忙碌着。这就...
安卓系统是什么操作系统,引领移... 亲爱的读者们,你是否曾在手机上看到过“安卓”这个词,却对它一知半解呢?今天,就让我带你一起揭开安卓系...
谷歌收购安卓1.0系统,开启移... 你有没有想过,现在我们手上的智能手机,曾经可是掀起了一场科技革命呢?没错,就是那个改变了我们生活的安...
鸿蒙系统和安卓游戏,畅享无缝跨... 亲爱的读者们,你是否也像我一样,对华为的鸿蒙系统和安卓游戏之间的兼容性充满好奇呢?今天,我就要带你深...
安卓系统开源字体设置,个性化与... 你有没有发现,手机上的字体有时候真的让人提不起精神?别急,今天就来教你怎么给安卓手机换上自己喜欢的开...
电视系统安卓易柚,康佳电视的智... 亲爱的读者们,你是否曾为家里的电视系统而烦恼?市面上那么多选择,到底哪个最适合你呢?今天,就让我带你...
修改安卓系统全局字体,安卓系统... 你有没有发现,手机里的字体有时候真的让人提不起精神?是不是也想给手机换换新装,让它看起来更有个性呢?...
安卓系统做电脑u盘系统,电脑U... 你有没有想过,用安卓系统来装电脑U盘?听起来是不是有点酷炫?没错,现在就有这么一个神奇的方法,让你不...
安卓 双核 双系统,探索安卓双... 你有没有想过,你的手机可以同时拥有两个不同的世界呢?没错,就是安卓双核双系统!想象一边是熟悉的安卓世...
安卓原生系统 充电模块,揭秘高... 你有没有发现,现在手机充电速度简直就像赛跑一样,快得让人眼花缭乱!不过,你知道吗?安卓原生系统在判定...
安卓系统换成苹果键盘,轻松切换... 亲爱的手机控们,是不是觉得安卓手机的键盘有点单调,而苹果手机的键盘又那么吸引人呢?别急,今天就来教你...
安卓系统适配键盘丝印,安卓系统... 你有没有发现,用安卓手机打字的时候,有时候键盘上的字母会变得模糊不清,甚至有时候还会出现错别字呢?这...
安卓系统双开封号,揭秘安卓系统... 安卓系统双开封号:揭秘背后的风险与机遇在数字化时代,手机已经成为我们生活中不可或缺的一部分。而在众多...
安卓11系统的nfc,Andr... 你有没有发现,现在手机的功能越来越强大了?尤其是安卓11系统的NFC功能,简直就像是个贴心的智能小助...
安卓系统怎么拍摄屏幕,轻松捕捉... 亲爱的手机控们,你是否有过这样的时刻:想要记录下手机屏幕上的精彩瞬间,却不知道怎么操作?别急,今天就...
安卓开发KTV点歌系统,基于安... 你有没有想过,在家就能享受到KTV的乐趣?现在,这不再是梦想啦!随着科技的发展,安卓开发KTV点歌系...
苹果手机安卓系统办公,智能手机... 你有没有发现,现在不管是上班还是出差,手机已经成了我们办公的小助手啦!尤其是苹果手机和安卓系统,它们...
字体怎么安装安卓系统,轻松实现... 你有没有想过,你的安卓手机字体是不是有点单调呢?想要给手机换上新的字体,让它看起来更有个性?别急,今...