闭包初理解

        当一个嵌套的内部函数(子)函数引用了嵌套的外部(父)函数的变量(函数)时,就产生了闭包,也可以理解为闭包是嵌套的内部函数或者是包含被引用变量(函数)的对象,那么闭包产生的条件是什么呢?一是需要有函数嵌套,二是要有内部函数引用外部函数的数据(变量/函数),两种条件具备就会产生闭包。

        闭包的作用有两个方面,一方面,使用函数内部的变量在函数执行完成后,仍然存活在内存中(延长了局部变量的生命周期),另一方面,让函数外部可以操作(读写)到函数内部的数据(变量/函数),前者其实也算是闭包的缺点,因为函数执行完成后,函数内的局部变量没有释放,占用内存时间会变长,容易造成内存泄露。要想解决这一问题,可以尽量不用闭包,或者在函数执行完成后及时释放。(如下面代码的最后一行,该数组占的空间是非常大的,所以应该在函数执行之后及时释放,将f赋值为null使内部函数成为垃圾对象,此操作即可回收闭包)

 function fn1(){
            var arr=new Array[10000];
            function fn2(){
                console.log(arr.length);
            }
            return fn2;
        }
        var f=fn1();
        f();
        f=null;//让内部函数成为垃圾对象-->回收闭包

        关于上面的代码,一开始我是有点看不懂的,但其实也是很好理解的。

        首先我们需要知道,常见的闭包有两种,一种是将函数作为另一个函数的返回值,另一种是将函数作为实参传递给另一个函数调用,上面的代码就是第一种,也就是说f指向函数对象,函数对象关联着闭包,用f把返回值存了起来,即f存着返回值fn2,那么f()就等于fn2(),也就是调用fn2函数。

Lua闭包与C#闭包在多个方面存在区别。在变量作用域上,C#的`for`循环里,变量通常在循环之声明,闭包捕获的是循环体外部的同一个变量;而Lua的`for`循环中,变量是在循环体内,每次都是新的局部变量,所以匿名函数中的变量是变化的。例如C#代码`for(int i = 0; i < 3; i++)`,闭包里实际都是循环体外部的`i`;而Lua的`for`循环每次循环的`i`都是独立的新局部变量[^2]。 在交互和类型转换方面,C#与Lua交互时,类型转换并非由Lua虚拟机完成,而是通过XLua/NLua等中间件实现桥接。基础类型(int/string/bool等)可直接映射,复杂类型(对象、容器等)会被包装为Lua的userdata并附加元表支持操作。Lua只能通过中间件提供的接口访问这些C#对象,无法直接理解其内部结构。自定义类型需注册后才能使用,否则在Lua中将显示为无效userdata或nil。整个过程本质是C#层的数据封装与代理调用,Lua侧仅获得操作句柄而非真实数据结构,C#闭包在这种交互场景下会涉及到这些复杂的类型转换和数据封装过程,而Lua闭包主要在自身的语言环境中运行,不存在这种与外部语言交互的类型转换机制[^3]。 在闭包实例和上值处理上,以Lua为例,不同闭包实例对局部变量的共享情况不同,如`f1`和`f2`分别创建闭包实例,`var1`是top level的局部变量能共享,`var2`则是独立上值,且在相关函数调用完毕后会从栈上移除,转为close状态;C#闭包在变量的共享和生命周期管理上遵循C#自身的内存管理和作用域规则,与Lua闭包在这方面的处理方式不同[^5]。 ```csharp // C#闭包示例 using System; class Program { static void Main() { Action[] actions = new Action[3]; for (int i = 0; i < 3; i++) { actions[i] = () => Console.WriteLine(i); } foreach (var action in actions) { action(); } } } ``` ```lua -- Lua闭包示例 local actions = {} for i = 1, 3 do actions[i] = function() print(i) end end for _, action in ipairs(actions) do action() end ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值