lua闭包解析

闭包的定义:

当一个函数(func)的返回值是另外一个函数(func内return的匿名函数),而返回的那个函数(func内return的匿名函数)调用了其父函数(func)内部的变量(x),且返回的这个函数在外部被执行了(print(f1())),就产生了闭包。闭包是一个环境,具体指的就是外部函数--高阶函数。

lua里面的函数,和c语言的函数其实是不同的概念。在lua里通常讲的函数,其实是指闭包(closure)。函数只是闭包的原形(prototype)声明。为了方便理解,所以通常讲函数。

lua里的函数是一个具有词法定界的第一类值。

第一类值定义:1.能够存储在变量中;2.存储在表中;3.能够作为函数的参数传递;4.能够作为函数的返回值。

词法定界:是指一个嵌套的函数(func内return的匿名函数)能够访问父函数(func)的变量(x)。

用处:1.读取函数内部的变量;

           2.这些变量的值始终保持在内存中,不会在外层函数调用后被自动清除。

优点:1:变量长期驻扎在内存中;

           2:避免全局变量的污染;

           3:私有成员的存在 ;

特性:1:函数套函数;

           2:内部函数可以直接使用外部函数的局部变量或参数;

           3:变量或参数不会被垃圾回收机制回收 GC;

缺点:

    常驻内存会增大内存的使用量使用不当会造成内存泄露,详解:

(1)由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE中可能导致内存泄露。解决方法是,在退出函数之前,将不使用的局部变量全部删除。

(2)闭包会在父函数外部,改变父函数内部变量的值。所以,如果你把父函数当作对象(object)使用,把闭包当作它的公用方法(Public Method),把内部变量当作它的私有属性(private value),这时一定要小心,不要随便改变父函数内部变量的值。

 

一个闭包的lua代码。(把下面代码拷贝到 https://www.lua.org/cgi-bin/demo 可以直接运行)

local y = 0
local function func()
    local x = 0
    return function ()
        x = x + 1
        return x + y
    end
end

local f1 = func()
local f2 = func()
print(f1()) --1
print(f1()) --2

print(f2()) --1
print(f2()) --2


y = 10
print(f1()) --13
print(f2()) --13

 

对于f1和f2,x,y都是upvalue。可以看出x是独立的upvalue,而y是共享的upvalue。
upvalue所指向的值,如果在栈中,即仍在作用域内,此upvalue叫做open upvalue。如果upvalue指向的值已经退栈,超出了作用域,则生成一份单独的拷贝upvalue,此upvalue叫做close upvalue。

lua解释器维护了一个open upvalue链表。当需要引用upvalue时,首先遍历此链表,实现upvalue的复用。如果没有找到,则在链表中插入此upvalue。当upvalue退出栈时,也会从open upvalue中删除。

当f1和f2调用时,x早已经退出栈,所以f1和f2会生成独立的upvalue--x,而对于y,还在栈中,因此是共用的upvalue。

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值