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

相关内容

热门资讯

安卓自己刷系统的软件,揭秘热门... 你有没有想过,你的安卓手机其实可以像换衣服一样,随时换上新的系统?没错,就是那种完全改变手机面貌的感...
cf分不分苹果系统和安卓系统,... 你有没有发现,现在手机市场上,苹果系统和安卓系统就像是一对“欢喜冤家”,总是被拿来比较。这不,最近就...
安卓最新系统版本壁纸,壁纸背后... 亲爱的手机控们,你是否已经迫不及待想要一睹安卓最新系统版本的芳容了呢?没错,今天我要带你一起探索这个...
怎样让安卓系统变成个s系统,轻... 你有没有想过,你的安卓手机能不能也来点个性,变成那个酷炫的S系统呢?别急,今天就来手把手教你,怎样让...
安卓平板如何双系统,打造个性化... 你有没有想过,你的安卓平板可以同时运行两个操作系统呢?没错,就是那种一个平板,两个世界的感觉!今天,...
忠诚卫士安装安卓系统 你有没有想过,你的忠诚卫士手机,是不是也能像电脑一样,装上各种有趣的安卓系统呢?没错,今天就要来给你...
安卓虚拟win系统教程,轻松实... 你有没有想过在安卓手机上也能体验Windows系统的感觉?没错,就是那种熟悉的桌面、任务栏,还有那些...
安卓原生系统谁做的,谁主导了这... 你有没有想过,手机里那个流畅又强大的安卓原生系统,究竟是谁的杰作呢?没错,今天就要揭开这个谜底,让你...
安卓系统小红书多开教程,轻松实... 你是不是也和我一样,对安卓系统的小红书多开功能感兴趣呢?想象一边刷着小红书,一边还能同时看几篇热门笔...
安卓导航怎么强制进系统,安卓导... 你是不是也遇到了这样的烦恼:手机里的安卓导航软件总是跳出来,让你无法专心开车或者导航?别急,今天就来...
手机虚拟安卓系统哪个好,手机虚... 你有没有想过,手机上的虚拟安卓系统就像是一扇通往新世界的门?想象你可以在同一部手机上体验不同的操作系...
安卓系统下载网站视频,轻松获取... 你有没有想过,手机里那些精彩纷呈的视频,其实都是通过安卓系统下载网站得来的?今天,就让我带你一探究竟...
原生安卓系统刷机包 亲爱的手机控们,你是否曾为手机系统的不够流畅而烦恼?或者,你只是想尝试一下原生安卓系统的纯粹魅力?那...
安卓系统自己增加内存,安卓系统... 你知道吗?最近在安卓系统里发现了一个超级实用的功能,那就是它竟然能自己增加内存!是不是听起来有点神奇...
venue11pro装安卓系统... 你有没有想过,你的 Venue 11 Pro 手机装上安卓系统后,会是怎样的一个场景呢?想象原本稳重...
原生安卓车机系统下载,打造智能... 车友们,你是否厌倦了那千篇一律的车载系统?想要给你的爱车换换“口味”?那就跟着我一起探索一下原生安卓...
通用制卡安卓系统版本 你有没有发现,手机里的那些卡片应用越来越智能了?这不,最近我在研究通用制卡安卓系统版本的时候,发现了...
安卓系统最好的省电王,助你续航... 手机电量总是不够用?是不是每次出门前都要检查一下充电宝在不在?别急,今天我要给你介绍一个安卓系统里的...
安卓通话系统变成黑色了,探究原... 最近你的安卓手机通话界面是不是突然变成了黑色?别急,别慌,让我来给你详细解析一下这个现象,让你轻松应...
长沙雨天攻略系统和安卓,畅游雨... 长沙这座美丽的城市,一到雨天就变得特别有味道。但你知道吗,雨天出行可不是件容易事儿,尤其是对于不熟悉...