一、基础知识:
1. 第一个程序和函数: 在目前这个学习阶段,运行Lua程序最好的方式就是通过Lua自带的解释器程序,如: /> lua > print("Hello World") Hello World 这样我们就可以以交互性的方式输入lua代码,并立即得到执行结果了。对于代码块较少的测试程序来说,这种方式确实是非常方便的,然而对于相对复杂的程序而言,这种方式就不是很合适了。如果是这样,我们可以将Lua代码保存到一个独立的Lua程序文件中,之后再通过Lua解释器程序以命令行参数的形式执行文件中的Lua代码。如我们将下面的Lua代码保存到test.lua的文件中:
1 function fact(n) 2 if n == 0 then 3 return 1 4 else 5 return n * fact(n - 1) 6 end 7 end 8 print("Enter a number:") 9 a = io.read("*number") 10 print(fact(a))
/> lua D:/test.lua Enter a number: 4 24
2. 代码规范: 1). Lua的多条语句之间并不要求任何分隔符,如C语言的分号(;),其中换行符也同样不能起到语句分隔的作用。因此下面的写法均是合法的。如:
1 a = 1 2 b = a * 2 3 4 a = 1; 5 b = a * 2; 6 7 a = 1; b = a * 2; 8 a = 1 b = a * 2
2). 通过dofile()方法引用其他Lua文件中的函数,如:
1 function fact(n) 2 if n == 0 then 3 return 1 4 else 5 return n * fact(n - 1) 6 end 7 end
将上面的函数保存到test2.lua文件中。 /> lua > dofile("d:/test2.lua") > print(fact(4)) 24 3). 词法规范。 和大多数其它语言一样,在声明变量时,变量名可以由任意字母、数字和下划线构成,但是不能以数字开头。在Lua中还有一个特殊的规则,即以下划线(_)开头,后面紧随多个大写字母(_VERSION),这些变量一般被Lua保留并用于特殊用途,因此我们在声明变量时需要尽量避免这样的声明方式,以免给后期的维护带来不必要的麻烦。 Lua是大小写敏感的,因此对于一些Lua保留关键字的使用要特别小心,如and。但是And和AND则不是Lua的保留字。 4). Lua中的注释分为两种,一种是单行注释,如: --This is a single line comment. 另外一种是多行注释,如: --[[ This is a multi-lines comment. --]]
3. 全局变量: 在Lua中全局变量不需要声明,直接赋值即可。如果直接访问未初始化的全局变量,Lua也不会报错,直接返回nil。如果不想再使用该全局变量,可直接将其置为nil。如: /> lua > print(b) nil > b = 10 > print(b) 10 > b = nil > print(b) nil 4. 解释器程序: 命令行用法如下: lua [options] [lua-script [arguments] ] 该工具的命令行选项主要有以下3个: -e: 可以直接执行命令行中Lua代码,如:lua -e "print(\"Hello World\")" -l: 加载该选项后的Lua库文件,如:lua -l mylib -e "x = 10",该命令在执行之前先将mylib中的Lua代码加载到内存中,在后面的命令中就可以直接使用该文件中定义的Lua函数了。 -i: 在执行完指定的Lua程序文件之后,并不退出解释器程序,而是直接进入该程序的交互模式。 在解释器程序的交互模式下,我们可以通过在表达式前加等号(=)标识符的方式直接输出表达式的执行结果。通过该方式,我们可以将该程序用于计算器,如: /> lua > = 3 + 1 + 4 8 该小节最后需要介绍的是lua脚本的命令行参数访问规则。如: /> lua lua-script.lua a b c 在该脚本的程序入口,lua解释器会将所有命令行参数创建一个名为arg的table。其中脚本名(lua-script.lua)位于table索引的0位置上。它的第一个参数(a)则位于索引1,其它的参数以此类推。这种索引方式和C语言中读取命令行参数的规则相同。但是不同的是,Lua提供了负数索引,用以访问脚本名称之前的命令行参数,如: arg[-1] = lua arg[0] = lua-script.lua arg[1] = a arg[2] = b arg[3] = c
二、类型与值:
Lua是一种动态类型的语言。其语言本身没有提供类型定义的语法,每个值都“携带”了它自身的类型信息。在Lua中有8中基础类型,分别是:nil、boolean、number、string、userdata、function、thread和table。我们可以通过type函数获得变量的类型信息,该类型信息将以字符串的形式返回。如: > print(type("hello world")) string > print(type(10.4)) number > print(type(print)) function > print(type(true)) boolean > print(type(nil)) nil > print(type(type(X))) string
1. nil(空): nil是一种类型,它只有一个值nil,它的主要功能是由于区别其他任何值。就像之前所说的,一个全局变量在第一次赋值前的默认值的默认值就是nil,将nil赋予一个全局变量等同于删除它。Lua将nil用于表示一种“无效值”的情况。 2. boolean(布尔): 该类型有两个可选值:false和true。在Lua中只有当值是false和nil时才视为“假”,其它值均视为真,如数字零和空字符串,这一点和C语言是不同的。 3. number(数字): Lua中的number用于表示实数。Lua中没有专门的类型表示整数。 4. string(字符串): Lua中的字符串通常表示“一个字符序列”。字符串类型的变量是不可变的,因此不能像C语言中那样直接修改字符串的某一个字符,而是在修改的同时创建了新的字符串。如:
1 a = "one string" 2 b = string.gsub(a,"one","another") 3 print(a) 4 print(b)
/> lua d:/test.lua one string anotner string Lua支持和C语言类似的字符转义序列,见下表:
转义符 |
描述 |
\a |
响铃 |
\b |
退格 |
\n |
换行 |
\r |
回车 |
\t |
水平Tab |
\\ |
反斜杠 |
\" |
双引号 |
\' |
单引号 |
在Lua中还可以通过[[ all strings ]]的方式来禁用[[ ]]中转义字符,如: page = [[ <html> <head> <title> An Html Page </title> </head> ]] 如果两个方括号中包含这样的内容:a = b[c[i]],这样将会导致Lua的误解析,因此在这种情况下,我们可以将其改为[===[ 和 ]===]的形式,从而避免了误解析的发生。 Lua提供了运行时的数字与字符串的自动转换。如: > print("10" + 1) 11 > print("10 + 1") 10 + 1 如果在实际编程中,不希望两个数字字符串被自动转换,而是实现字符串之间的连接,可以通过" .. "操作符来完成。如: > print(10 .. 20) 1020 注意..和两边的数字之间必须留有空格,否则就会被Lua误解析为小数点儿。 尽管Lua提供了这种自动转换的功能,为了避免一些不可预测的行为发生,特别是因为Lua版本升级而导致的行为不一致现象。鉴于此,还是应该尽可能使用显示的转换,如字符串转数字的函数tonumber(),或者是数字转字符串的函数tostring()。对于前者,如果函数参数不能转换为数字,该函数返回nil。如:
1 line = io.read() 2 n = tonumber(line) 3 if n == nil then 4 error(line .. " is not a valid number") 5 else 6 print(n * 2) 7 end
关于Lua的字符串最后需要介绍的是"#"标识符,该标识符在字符串变量的前面将返回其后字符串的长度,如:
/> lua d:/test.lua 5 5. table(表): 我们可以将Lua中table类型视为“关联数组”,如C++标准库中的map,差别是Lua中table的键(key)可以为任意类型(nil除外),而map中的键只能为模参类型。此外,table没有固定的大小,可以动态的添加任意数量的元素到一个table中。table是Lua中最主要数据结构,其功能非常强大,可用于实现数组、集合、记录和队列数据结构。以下为table的变量声明,以及关联数据的初始化方式:
1 a = {} -- 创建一个table对象,并将它的引用存储到a 2 k = "x" 3 a[k] = 10 -- 创建了新条目,key = "x", value = 10 4 a[20] = "great" -- 新条目,key = 20, value = "great" 5 print(a["x"]) 6 k = 20 7 print(a[k]) -- 打印great 8 a["x"] = a["x"] + 1 9 print(a["x"]) -- 打印11
所有的table都可以用不同类型的索引来访问value,当需要容纳新条目时,table会自动增长。
1 a = {} 2 for i = 1, 100 do 3 a[i] = i * 2 4 end 5 print(a[9]) 6 a["x"] = 10 7 print(a["x"]) 8 print(a["y"]) --table中的变量和全局变量一样,没有赋值之前均为nil。 9 10 --输出结果为 11 --18 12 --10 13 --nil
在Lua中还提供了另外一种方法用于访问table中的值,见如下示例:
1 a.x = 10 --等同于a["x"] = 10 2 print(a.x) --等同于print(a["x"]) 3 print(a.y) --等同于print(a["y"])
对于Lua来说,这两种方式是等价的。但是对于开发者而言,点的写法隐式的将table表示为记录,既C语言中的结构体。而之前讲述的字符串表示法则意味着任何字符串均可作为table的key。 如果需要将table表示为传统的数组,只需将整数作为table的key即可。如:
1 a = {} 2 for i = 1,10 do 3 a[i] = i * 2 4 end 5 6 for i = 1,10 do 7 print(a[i]) 8 end
在Lua中,我通常习惯以1作为数组索引的起始值。而且还有不少内部机制依赖于这个惯例。如:
1 a = {} 2 for i = 1,10 do 3 a[i] = i * 2 4 end 5 6 for i = 1,#a do 7 print(a[i]) 8 end
由于数组实际上仍为一个table,所以对于数组大小的计算需要留意某些特殊的场景,如: a = {} a[1000] = 1 在上面的示例中,数组a中索引值为1--999的元素的值均为nil。而Lua则将nil作为界定数据结尾的标志。当一个数组含有“空隙”时,即中间含有nil值,长度操作符#会认为这些nil元素就是结尾标志。当然这肯定不是我们想要的结果。因此对于这些含有“空隙”的数组,我们可以通过函数table.maxn()返回table的最大正数索引值。如:
1 a = {} 2 a[1000] = 1 3 print(table.maxn(a)) 4 5 -- 输出1000
6. function(函数): 在Lua中,函数可以存储在变量中,可以通过参数传递其它函数,还可以作为其它函数的返回值。这种特性使语言具有了极大的灵活性。
7. userdata(自定义类型): 由于userdata类型可以将任意C语言数据存储到Lua变量中。在Lua中,这种类型没有太多预定义的操作,只能进行赋值和相等性测试。userdata用于表示一种由应用程序或C语言库所创建的新类型。
|
请发表评论