Lua函数以及闭合函数的理解
Lua函数以及闭合函数的理解
来源 http://blog.csdn.net/mydad353193052/article/details/48731467
词法域和第一类型
在C/C++,C#或者Java等传统语言中,一个函数A,如果想调用另一个函数B,并且B需要访问A中的变量,那么A就需要向B传递参数,参数形式可以是普通类型,指针,或者引用。(C#中有out输出参数和ref引用,专门的关键字来做这件事)但是在Lua中,呵呵,不必如此。
Lua中有一个“词法域”的概念。即B可以访问他所需要访问的所有“非局部变量”(后面会解释为什么叫双引号下的非局部变量)。但是这一切的说说笑笑都必须要有前提——在Lua中,函数是“第一类型”。就是说函数和int,string,float等泛泛之辈一样,都TM是类型,都是菜。而函数名则可以理解为,拿着函数实体的变量。请看下面这个例子:
- a = {p = print}
- a.p("hello bitch") ---hello bitch
- print = math.sin ---'print'现在是正弦函数
- a.p(print(1)) ---0.8414709848079
- sin = a.p ---'sin'现在是print函数
- sin(10 .. 20) ---1020
正所谓似鸡非鸡,似鸭非鸭。鸡有可能变鸭,鸭也可以手术变鸡。
那么说函数是变量,他还就蹬鼻子上脸,给变量一会看看——Lua常见的函数编写方式:
function foo(x) return 2*x end
其实他是一下这句话的简化形式:
foo = function (x) return 2*x end
所以,一个函数的定义实际就是一条赋值语句,这条语句首先创建一个函数类型的值,然后将这个值赋予这个函数变量。也可以将表达式 function(x)<body>end 理解为一种函数的构造式。就像table的构造式 {} 一样。
将这种函数构造式的结果称为一个“匿名函数”。下面这个例子就是使用匿名函数来做参数。
table.sort,是对一个table表中的内容按照一定规则排序。但是sort并没有提供具体的规则,规则的指定完全交给使用者自己定义。升序,降序,按key顺序等等。sort接收两个元素,第一个元素是table,第二个元素就是排序规则。那我们在这里就用匿名函数来做这个sort的第二个元素——排序规则。
- network = {
- {name = "grauna", NICK = "lululala"},
- {name = "araadd", NICK = "bababu" },
- {name = "oo", NICK = "ppreea"},
- {name = "zzdafae", NICK = "ddadf"},
- }
- for i=1, #network do
- print(network[i].name)
- end
- table.sort(
- network,
- function(a,b)
- return (a.name > b.name)
- end
- )
- for k,v in pairs(network) do
- print(k,v.name)
- end
二、闭合函数
在理解了上面这些思想之后,我们来想想closure。还是刚才说的, A==>>B,A是B的外部函数,B可以访问A中的局部变量。直接上菜说:
- names = {"Peter", "Paul", "Mary"}
- grades = {Mary = 10, Paul = 7, Peter = 8}
- table.sort(
- names,
- function(n1, n2)
- return grades[n1] > grades[n2]
- end
- )
- for k,v in ipairs(names) do
- print(k,v)
- end
用一个单独的函数来调用:
- names = {"Peter", "Paul", "Mary"}
- grades = {Mary = 10, Paul = 7, Peter = 8}
- function sortbygrade(cnames, cgrades)
- table.sort(
- cnames,
- function(n1, n2)
- return cgrades[n1] > cgrades[n2]
- end
- )
- return cnames
- end
- for k,v in ipairs(sortbygrade(names, grades)) do
- print(k,v)
- end
输出结果同上。
可见,传递给sort的匿名函数可以访问参数cgrades,而cgrades是外部函数sortbygrade的局部变量。而在这个匿名函数内部,cgrades既不是全局变量也不是局部变量,将其称为一个“非局部的变量”。
再看下面这个例子:
- function newCounter()
- local i = 0
- return function()
- i = i + 1
- return i
- end
- end
- c1 = newCounter()
- print(c1()) ---1
- print(c1()) ---2
- c2 = newCounter()
- print(c2()) ---1
- print(c1()) ---3
- print(c2()) ---2
初次见到这个输出结果,有些摸不到后脑勺。为啥第二次print(c1())结果是2呢?
仔细一想,这还得说那句老话,Lua中函数是第一类型。(别骂娘)
你想想,在C++中,如果你定义一个int a=1;那么编译器只会在栈中开辟一块空间来存a,直至释放。
那么c1也是这回事儿,你第二次print,实际上还是在 第一次在栈中开辟的那块空间里 折腾。所以此时栈中的local i已经不是当年那个 = 0 的小姑娘了,人家已经长成 1 了。所以你再print,就在1的基础上变成2了。
而c2跟c1不一样,c2是隔壁家的姑娘,又是一块栈空间,所以c2的第一次还是1。 以此类推,后面的print结果就顺理成章了。
Lua函数以及闭合函数的理解的更多相关文章
- lua闭合函数
function count( ... ) return function( ... ) i = i+ return i end end local func = count(...) print(f ...
- [Lua] 迭代器 闭合函数 与 泛型for
首先看看一下闭合函数(closure),见如下代码: function newCounter() local i = 0 -- 非局部变量(non-local variable) return fun ...
- Lua中的closure(闭合函数)
词法域:若将一个函数写在另一个函数之内,那么这个位于内部的函数便可以访问外部函数中的局部变量,这项特征称之为“词法域”. 例:假设有一个学生姓名的列表和一个对应于没个姓名的年级列表,需要根据每个学生的 ...
- Lua中调用C函数
Lua利用一个虚拟的堆栈来给C传递值或从C获取值.每当Lua调用C函数,都会获得一个新的堆栈,该堆栈初始包含所有的调用C函数所需要的参数值(Lua传给C函数的调用实参),并且C函数执行完毕后,会把返回 ...
- lua学习之深入函数第一篇
深入函数第一篇 函数是第一类值,具有特定的词法域 第一类值 第一类值的意思是函数与 lua 中的其他类型如数字,字符串具有相同的权力 函数可以存储到全局变量或局部变量变量,还可以存储到 table 中 ...
- 深入理解javascript函数定义与函数作用域
最近在学习javascript的函数,函数是javascript的一等对象,想要学好javascript,就必须深刻理解函数.本人把思路整理成文章,一是为了加深自己函数的理解,二是给读者提供学习的途径 ...
- lua如何调用C++函数
第一步是定义函数.所有在Lua中被调用的C/C++函数将使用下面一类指针进行调用: typedef int (*lua_CFunction) (lua_State *L); 换句话说,函数必须要以Lu ...
- Lua中的常用函数库汇总
lua库函数 这些函数都是Lua编程语言的一部分, 点击这里了解更多. assert(value) - 检查一个值是否为非nil, 若不是则(如果在wow.exe打开调试命令)显示对话框以及输出错误调 ...
- lua堆栈操作常用函数学习二
/* ** basic stack manipulation */ LUA_API int <strong> (lua_gettop) (lua_State *L); </str ...
随机推荐
- 逆向实战干货,快速定位自动捡阳光Call,或者标志
逆向实战干货,快速定位自动捡阳光Call,或者标志 注意: 关于CE和OD的使用,这里不再多说,快速定位,默认大家已经有了CE基础,或者OD基础. 第一种方法,找Call 第一步,打开CE,搜索阳光值 ...
- 实例讲解webpack的基本使用第四篇
这一篇来讲解一下webpack的loader的使用,用webpack打包文件,css,img,icon等都需要下载安装对应的loader文件,并且写好配置项,才可以进行打包,废话不多说,直接开始实战. ...
- js中set和get的用法
get 语句作为函数绑定在对象的属性上,当访问该属性时调用该函数. set 语法可以将一个函数绑定在当前对象的指定属性上,当那个属性被赋值时,你所绑定的函数就会被调用. eg: var log = [ ...
- php根据ip段以及子网掩码,判断某ip是否处于某子网下
为了检测客户端ip是否位于指定的网络里(如防火墙过滤有时候需要用到这个技术),有如下方法: 1.第一种 public function netMatch($client_ip, $server ...
- 实现mysql在windows server 2008下自动备份
环境:MySQL 安装位置:D:\MySQL论坛数据库名称为:Db_Test数据库备份目的地:D:\db_bak\ 1.首先新建一个bat文件 rem ********************** ...
- (原)怎样解决python dataframe loc,iloc循环处理速度很慢的问题
怎样解决python dataframe loc,iloc循环处理速度很慢的问题 1.问题说明 最近用DataFrame做大数据 处理,发现处理速度特别慢,追究原因,发现是循环处理时,loc,iloc ...
- Javascript从“繁”到“简”进行数组去重
随着JavaScript提供语法的增多,数组去重方式也越来越多.现在从最原始的方式到最简洁的方式,一步步进行剖析. 双重循环 数组去重,不就是比较数组元素,去掉重复出现的么.最原始的方式不正是双重循环 ...
- 我的第一个python web开发框架(7)——本地部署前端访问服务器
PS:本系列内容进度节奏会放的很慢,每次知识点都尽量少一点,这样大家接触的知识点少了,会更容易理解,因为少即是多.另外,对于后面代码部分,虽然尽量不用那些复杂的封装和类,但它并不表示看了就能全部记住, ...
- 【源码】canal和otter的高可靠性分析
一般来说,我们对于数据库最主要的要求就是:数据不丢.不管是主从复制,还是使用类似otter+canal这样的数据库同步方案,我们最基本的需求是,在数据不丢失的前提下,尽可能的保证系统的高可用,也就是在 ...
- 【转载】使用CSS将图片转换成黑白(灰色、置灰)
文章转载自 张鑫旭-鑫空间-鑫生活 http://www.zhangxinxu.com/ 原文链接:http://www.zhangxinxu.com/wordpress/?p=2547原文摘要: . ...