码云代码链接
https://gitee.com/wenwenc9/lua_pro.git
table 是 Lua 的一种数据结构用来帮助我们创建不同的数据类型,如:数字、字典等。
Lua table 使用关联型数组,你可以用任意类型的值来作数组的索引,但这个值不能是 nil。
Lua table 是不固定大小的,你可以根据自己需要进行扩容。
Lua也是通过table来解决模块(module)、包(package)和对象(Object)的。
例如string.format表示使用"format"来索引table string。
#table 获取表的长度
一、表的构造
构造器是创建和初始化表的表达式。表是Lua特有的功能强大的东西。最简单的构造函数是{},用来创建一个空表。可以直接初始化数组:
-- 初始化表
mytable = {}
-- 指定值
mytable[1]= "Lua"
-- 移除引用
mytable = nil
-- lua 垃圾回收会释放内存
构造函数可以使用任何表达式初始化
tab = {math.max(5,2),math.max(4,7)}
如果想初始化一个表作为 record 使用可以这样:
a = {x=0, y=0} <–> a = {}; a.x=0; a.y=0
不管用何种方式创建 table,我们都可以向表中添加或者删除任何类型的域,构造函
数仅仅影响表的初始化。
复杂构造,表中嵌套表调用
w = {x=0, y=0, label="console"}
x = {math.sin(0), math.sin(1), math.sin(2)}
w[1] = "another field"
x.f = w
print(w["x"]) -- 0
print(w[1]) -- another field
print(x.f) -- table: 00AE9E50
print(x.f[1]) -- another field
print(x.f[2]) -- nil
print(x.f['label']) -- console
print(x[1]) -- 0
print(x[2]) -- 0.8414709848079
print(x[3]) -- 0.90929742682568
w.x = nil -- remove field "x"
每次调用构造函数,Lua 都会创建一个新的 table,可以使用 table 构造一个 list:
list = nil
for line in io.lines() do
list = {next=list, value=line}
end
这段代码从标准输入读进每行,然后反序形成链表。下面的代码打印链表的内容
l = list
while l do
print(l.value)
l = l.next
end
在同一个构造函数中可以混合列表风格和 record 风格进行初始化,如:
polyline = { color = "blue", thickness = 2, npoints = 4,
{ x = 0, y = 0 },
{ x = -10, y = 0 },
{ x = -10, y = 1 },
{ x = 0, y = 1 }
}
for k,v in ipairs(polyline) do
print(k,':',v)
end
print('----------------------------------------------')
for k,v in next,polyline do
print(k,':',v)
end
print('----------------------------------------------')
for k,v in pairs(polyline) do
print(k,':',v)
end
print(polyline[2].x) -- -10
1 : table: 00C19CE8
2 : table: 00C19D38
3 : table: 00C19C20
4 : table: 00C19D60
----------------------------------------------
1 : table: 00C19CE8
2 : table: 00C19D38
3 : table: 00C19C20
4 : table: 00C19D60
color : blue
npoints : 4
thickness : 2
----------------------------------------------
1 : table: 00C19CE8
2 : table: 00C19D38
3 : table: 00C19C20
4 : table: 00C19D60
color : blue
npoints : 4
thickness : 2
上面两种构造函数的初始化方式还有限制,比如你不能使用负索引初始化一个表中
元素,字符串索引也不能被恰当的表示。下面介绍一种更一般的初始化方式,我们用
[expression]显示的表示将被初始化的索引:
opnames = {["+"] = "add", ["-"] = "sub",
["*"] = "mul", ["/"] = "div"}
list 风格初始化和 record 风格初始
化是这种一般初始化的特例:
{x=0, y=0} <--> {["x"]=0, ["y"]=0}
{"red", "green", "blue"} <-->
{[1]="red", [2]="green", [3]="blue"}
如果真的想要数组下标从 0 开始:
days = {[0]="Sunday", "Monday", "Tuesday", "Wednesday",
"Thursday", "Friday", "Saturday"}
注意:不推荐数组下标从 0 开始,否则很多标准库不能使用。
在构造函数的最后的","是可选的,可以方便以后的扩展。
a = {[1]="red", [2]="green", [3]="blue",}
print(a[1]) -- red
在构造函数中域分隔符逗号(",")可以用分号(";")替代,通常我们使用分号用来
分割不同类型的表元素。
{x=10, y=45; "one", "two", "three"}
当我们为 table a 并设置元素,然后将 a 赋值给 b,则 a 与 b 都指向同一个内存
。
如果 a 设置为 nil ,则 b 同样能访问 table 的元素。
如果没有指定的变量指向a,Lua的垃圾回收机制会清理相对应的内存。
二、表的操作
- table.concat() 表连接
- table.insert() 表插入
- table.maxn() 表最大值 5.2版本后移除
- table.remove() 表的删除
- table.sort() 排序
1、连接 concat
tab1 = {'apple','orange','abc'}
-- 全连接
print(table.concat(tab1)) -- appleorangeabc
-- 指定字符连接
print(table.concat(tab1,';')) -- apple;orange;abc
-- 指定索引连接 开始截止位
print(table.concat(tab1,';',1,3)) -- apple;orange;abc
2、插入和移除 insert/remove
插入和移除 默认都是 末尾
fruits = {"banana","orange","apple"}
-- 末尾插入
table.insert(fruits,"mango")
print(fruits[4]) -- mongo
-- 在索引为2的键处插入
table.insert(fruits,2,"grapes")
print(fruits[2]) -- grapes
print(fruits[5]) -- mongo
-- 移除
table.remove(fruits)
print(fruits[5])
3、排序 sort
fruits = {"banana","orange","apple","grapes"}
print("排序前")
for k,v in ipairs(fruits) do
print(k,v)
end
table.sort(fruits)
print("排序后")
for k,v in ipairs(fruits) do
print(k,v)
end
4、最大值
table.maxn 在 Lua5.2 之后该方法已经不存在了,我们定义了 table_maxn 方法来实现。
function table_maxn(t)
local mn = 0
for k, v in pairs(t) do
if mn < k then
mn = k
end
end
return mn
end
tbl = {[1] = "a", [2] = "b", [3] = "c", [26] = "z"}
print("tbl 长度 ", #tbl)
print("tbl 最大值 ", table_maxn(tbl))