zoukankan      html  css  js  c++  java
  • Lua的闭包详解(终于搞懂了)

    词法定界:当一个函数内嵌套另一个函数的时候,内函数可以访问外部函数的局部变量,这种特征叫做词法定界

        table.sort(names,functin (n1,n2)
            return grades[n1]>grades[n2]
        end)
        //内部匿名函数可以访问外部函数的n1,n2

    第一类值:lua当中函数是一个值,他可以存在变量中,可以作为函数参数,可以作为返回值

    复制代码
        function test()
            local i=0
            return function()
                i++
                ...
            end
        end
        //函数作为返回值,这里的i也叫外部局部变量,就是lua中的upvalue
    复制代码

    闭包:通过调用含有一个内部函数加上该外部函数持有的外部局部变量(upvalue)的外部函数(就是工厂)产生的一个实例函数

    闭包组成:外部函数+外部函数创建的upvalue+内部函数(闭包函数)

    实例:

    复制代码
        function test()
            local i=0
            return function()//尾调用
                i+=1
                return i
            end
        end
        c1=test()
        c2=test()//c1,c2是建立在同一个函数,同一个局部变量的不同实例上面的两个不同的闭包
                //闭包中的upvalue各自独立,调用一次test()就会产生一个新的闭包
        print(c1()) -->1
        print(c1()) -->2//重复调用时每一个调用都会记住上一次调用后的值,就是说i=1了已经
        print(c2())    -->1//闭包不同所以upvalue不同    
        print(c2()) -->2
    复制代码

    闭包在迭代器中的运用:迭代器需要保留上一次调用的状态和下一次成功调用的状态,刚好可以使用闭包的机制来实现

    创建迭代器:(一定要注意迭代器只是一个生成器,他自己本身不带循环)

    复制代码
         function list_iter(t)
                local i=0
                local n=table.getn(t)
                return function()
                    i=i+1
                    if i<=n then return t[i] end
                end
            end
        //这里的list_iter是一个工厂,每次调用都会产生一个新的闭包该闭包内部包括了upvalue(t,i,n)
        //因此每调用一次该函数产生的闭包那么该闭包就会根据记录上一次的状态,以及返回list的下一个
    复制代码

    使用迭代器:

    复制代码
            while中使用:
                t={10,20,90}
                iter=list_iter(t)//调用迭代器产生一个闭包
                while true do
                    local element=iter()
                    if element==nil then break end
                    print(element)
                    end
                end
            泛型for使用:
                t={10,0,29}
                for element in list_iter(t) do//这里的list_iter()工厂函数只会被调用一次产生一个闭包函数,后面的每一次迭代都是用该闭包函数,而不是工厂函数
                    print(element)
                end
    复制代码
  • 相关阅读:
    msyql数据库位置
    linux端口
    crontab
    floyd算法 青云的机房组网方案(简单)
    拓扑排序 codevs 4040 cojs 438
    高精度模板
    莫比乌斯函数
    二分算法~~~大综合
    莫比乌斯反演 BZOJ 2820
    2016.6.2考试整理
  • 原文地址:https://www.cnblogs.com/lancidie/p/8637609.html
Copyright © 2011-2022 走看看