发布于 2016-01-23 09:28:05 | 107 次阅读 | 评论: 0 | 来源: 网友投递
Lua 脚本语言
Lua 是一个小巧的脚本语言。是巴西里约热内卢天主教大学(Pontifical Catholic University of Rio de Janeiro)里的一个研究小组,由Roberto Ierusalimschy、Waldemar Celes 和 Luiz Henrique de Figueiredo所组成并于1993年开发。 其设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能。Lua由标准C编写而成,几乎在所有操作系统和平台上都可以编译,运行。Lua并没有提供强大的库,这是由它的定位决定的。所以Lua不适合作为开发独立应用程序的语言。Lua 有一个同时进行的GIT项目,提供在特定平台上的即时编译功能。
前言
Lua中的函数和C++中的函数的含义是一致的,Lua中的函数格式如下:
function MyFunc(param)
-- Do something
end
print "Hello World" --> print("Hello World")等价
print [[a multi-line
message]] -->print([[a multi-line
--> message]]) 等价
-- f是一个函数
f{x=10, y=20} -->f({x=10, y=20}) 等价
一个函数定义具有一个名称、一系列的参数和一个函数体。函数定义时,所定义的参数的使用方式与局部变量非常相似,它们是由调用函数时的“实际参数”初始化的。调用函数时提供的实参数量可以与形参数量不同。Lua会自动调整实参的数量,以匹配参数表的要求,若“实参多余形参,则舍弃多余的实参;若实参不足,则多余的形参初始化为nil”。这个与接下来要介绍的多重返回值非常相似。
多重返回值
这个应该是Lua的一个特征吧。允许函数返回多个结果,只需要在return关键字后列出所有的返回值即可。以下根据带来来说明情况:
function foo0() end -- 无返回值
function foo1() return "a" end -- 返回一个结果
function foo2() return "a", "b" end -- 返回两个结果
-- 在多重赋值时,如果一个函数调用是最后,或仅有的一个表达式,
-- 那么Lua会保留其尽可能多的返回值,用于匹配赋值变量
x, y = foo2() -- x = "a", y = "b"
x = foo2() -- x = "a", "b"被丢弃
x, y, z = 10, foo2() -- x = 10, y = "a", z = "b"
-- 如果一个函数没有返回值或者没有足够多的返回值,那么Lua会用
-- nil来补充缺失的值
x, y = foo0() -- x = nil, y = nil
x, y = foo1() -- x = "a", y = nil
x, y, z = foo2() -- x = "a", y = "b", z = nil
-- 如果一个函数调用不是一系列表达式的最后一个元素,那么将只产生一个值:
x, y = foo2(), 20 -- x = "a", y = 20
x, y = foo0(), 20, 30 -- x = nil, y = 20, 30则被丢弃
-- table构造式可以完整的接收一个函数调用的所有结果,即不会有任何数量
-- 方面的调整
local t = {foo0()} -- t = {}(一个空的table)
local t = {foo1()} -- t = {"a"}
local t = {foo2()} -- t = {"a", "b"}
-- 但是,对于上述的行为,只有当一个函数调用作为最后一个元素时才会发生,
-- 而在其他位置上的函数调用总是只产生一个结果值
local t = {foo0(), foo2(), 4} -- t[1] = nil, t[2] = "a", t[3] = 4
-- 我们也可以在一个函数中,使用return返回另一个函数
function MyFunc() -- 返回a
return foo1() -- 注:这里是return foo1(),而不是return (foo1())
end
-- return foo1()和return (foo1())是两个完全不同的意思
-- 将一个函数调用放入一对圆括号中,从而迫使它只返回一个结果
print((foo0())) -- nil
print((foo1())) -- a
print((foo2())) -- a
变长参数
在C语言中,函数可以接受不同数量的实参,Lua中的函数也可以接受不同数量的实参,例如以下代码:
-- 打印所有的参数
function VarArguments(...)
for i, v in ipairs{...} do
print(v)
end
end
VarArguments(1, 2, 3)
通常一个函数在遍历其变长参数时只需要使用表达式{…},这就像访问一个table一样,访问所有的变长参数。然而在某些特殊的情况下,变长参数中可能会包含一些故意传入的nil,那么此时就需要用select来访问变长参数了。调用select时,必须传入一个固定实参selector和一系列变长参数。如果selector为数字n,那么select返回它的第n个可变实参;否则selector只能为字符串“#”,这样select会返回变长参数的总数,请看以下代码:
for i = 1, select('#', ...) do
local arg = select(i, ...) -- 得到第i个参数
-- Do something else
end
function MyFunc(a, b, ...)
print(arg.n)
end
MyFunc(1, 2, 3, 4, 5) -->3
深入讨论函数
在Lua中,函数与其它传统类型的值具有相同的权利。函数可以存储到变量或table中,也可以作为实参传递给其它函数,还可以作为其它函数的返回值。在Lua中有一个容易混淆的概念是,函数与所有其它值一样都是匿名的,即它们都没有名称。当讨论一个函数名时,实际上是在讨论一个持有某函数的变量,例如以下代码:
-- 我们经常这样定义函数
function foo(x) return 2 * x end
-- 实际上,这只是一种“语法糖”而已;
-- 上述代码只是下面代码的一种简化书写形式
foo = function (x) return 2 * x end
内嵌函数
若将一个函数写在另一个函数之内,那么这个位于内部的函数便可以访问外部函数中的局部变量,这个特征叫做“词法域”。我们来看看下面一段有趣的代码:
function newCounter()
local i = 0
return function () -- 匿名函数
i = i + 1
return i
end
end
c1 = newCounter()
print(c1()) -->输出什么?
print(c1()) -->又输出什么?
function newCounter(i)
return function () -- 匿名函数
i = i + 1
return i
end
end
c1 = newCounter(10)
print(c1()) -->输出什么?
print(c1()) -->又输出什么?
非全局的函数
由于函数和普通变量一样,所以函数不仅可以存储在全局变量中,还可以存储在table的字段中,或局部变量中。我们可以把函数存在一个table中,比如以下代码:
Lib = {}
Lib.foo = function (x, y) return x + y end
Lib.goo = function (x, y) return x - y end
local f = function (<参数>)
<函数体>
end
-- Lua还提供另一种特殊的“语法糖”
local function f (<参数>)
<函数体>
end
local f, g
function f()
<一些其它操作>
g()
end
function g()
<一些其它操作>
f()
end
总结
这篇博文对Lua中的函数进行了大体上的总结,至少看完这篇博文,你会使用Lua写函数了,会使用Lua中的函数了。但是对于比较深的东西,这里没有总结,比如“闭包”。我会专门写一篇关于Lua中的闭包的文章。