Lua table(表)

table 是 Lua 的一种数据结构用来帮助我们创建不同的数据类型,如:数组、字典等。

Lua table 使用关联型数组,你可以用任意类型的值来作数组的索引,但这个值不能是 nil。

Lua table 是不固定大小的,你可以根据自己需要进行扩容。

Lua也是通过table来解决模块(module)、包(package)和对象(Object)的。例如string.format表示使用"format"来索引table string。

table(表)的构造

构造器是创建和初始化表的表达式。表是Lua特有的功能强大的东西。最简单的构造函数是{},用来创建一个空表。可以直接初始化数组:

  1. -- 初始化表
  2. mytable = {}
  3. -- 指定值
  4. mytable[1]= "Lua"
  5. -- 移除引用
  6. mytable = nil
  7. -- lua 垃圾回收会释放内存

当我们为 table a 并设置元素,然后将 a 赋值给 b,则 a 与 b 都指向同一个内存。如果 a 设置为 nil,则 b 同样能访问 table 的元素。如果没有指定的变量指向a,Lua的垃圾回收机制会清理相对应的内存。

以下实例演示了以上的描述情况:

实例

  1. -- 简单的 table
  2. mytable = {}
  3. print("mytable 的类型是 ",type(mytable))
  4. mytable[1]= "Lua"
  5. mytable["wow"] = "修改前"
  6. print("mytable 索引为 1 的元素是 ", mytable[1])
  7. print("mytable 索引为 wow 的元素是 ", mytable["wow"])
  8. -- alternatetablemytable的是指同一个 table
  9. alternatetable = mytable
  10. print("alternatetable 索引为 1 的元素是 ", alternatetable[1])
  11. print("mytable 索引为 wow 的元素是 ", alternatetable["wow"])
  12. alternatetable["wow"] = "修改后"
  13. print("mytable 索引为 wow 的元素是 ", mytable["wow"])
  14. -- 释放变量
  15. alternatetable = nil
  16. print("alternatetable 是 ", alternatetable)
  17. -- mytable 仍然可以访问
  18. print("mytable 索引为 wow 的元素是 ", mytable["wow"])
  19. mytable = nil
  20. print("mytable 是 ", mytable)

以上代码执行结果为:

  1. mytable 的类型是 table
  2. mytable 索引为 1 的元素是 Lua
  3. mytable 索引为 wow 的元素是 修改前
  4. alternatetable 索引为 1 的元素是 Lua
  5. mytable 索引为 wow 的元素是 修改前
  6. mytable 索引为 wow 的元素是 修改后
  7. alternatetable nil
  8. mytable 索引为 wow 的元素是 修改后
  9. mytable nil

Table 操作

以下列出了 Table 操作常用的方法:

序号方法 & 用途
1table.concat (table [, sep [, start [, end]]]): concat是concatenate(连锁, 连接)的缩写. table.concat()函数列出参数中指定table的数组部分从start位置到end位置的所有元素, 元素间以指定的分隔符(sep)隔开。
2table.insert (table, [pos,] value): 在table的数组部分指定位置(pos)插入值为value的一个元素. pos参数可选, 默认为数组部分末尾.
3table.maxn (table) 指定table中所有正数key值中最大的key值. 如果不存在key值为正数的元素, 则返回0。(Lua5.2之后该方法已经不存在了,本文使用了自定义函数实现)
4table.remove (table [, pos]) 返回table数组部分位于pos位置的元素. 其后的元素会被前移. pos参数可选, 默认为table长度, 即从最后一个元素删起。
5table.sort (table [, comp]) 对给定的table进行升序排序。

接下来我们来看下这几个方法的实例。

Table 连接

我们可以使用 concat() 输出一个列表中元素连接成的字符串:

实例

  1. fruits = {"banana","orange","apple"}
  2. -- 返回 table 连接后的字符串
  3. print("连接后的字符串 ",table.concat(fruits))
  4. -- 指定连接字符
  5. print("连接后的字符串 ",table.concat(fruits,", "))
  6. -- 指定索引来连接 table
  7. print("连接后的字符串 ",table.concat(fruits,", ", 2,3))

执行以上代码输出结果为:

  1. 连接后的字符串 bananaorangeapple
  2. 连接后的字符串 banana, orange, apple
  3. 连接后的字符串 orange, apple

插入和移除

以下实例演示了 table 的插入和移除操作:

实例

  1. fruits = {"banana","orange","apple"}
  2. -- 在末尾插入
  3. table.insert(fruits,"mango")
  4. print("索引为 4 的元素为 ",fruits[4])
  5. -- 在索引为 2 的键处插入
  6. table.insert(fruits,2,"grapes")
  7. print("索引为 2 的元素为 ",fruits[2])
  8. print("最后一个元素为 ",fruits[5])
  9. table.remove(fruits)
  10. print("移除后最后一个元素为 ",fruits[5])

执行以上代码输出结果为:

  1. 索引为 4 的元素为 mango
  2. 索引为 2 的元素为 grapes
  3. 最后一个元素为 mango
  4. 移除后最后一个元素为 nil

Table 排序

以下实例演示了 sort() 方法的使用,用于对 Table 进行排序:

实例

  1. fruits = {"banana","orange","apple","grapes"}
  2. print("排序前")
  3. for k,v in ipairs(fruits) do
  4. print(k,v)
  5. end
  6. table.sort(fruits)
  7. print("排序后")
  8. for k,v in ipairs(fruits) do
  9. print(k,v)
  10. end

执行以上代码输出结果为:

  1. 排序前
  2. 1 banana
  3. 2 orange
  4. 3 apple
  5. 4 grapes
  6. 排序后
  7. 1 apple
  8. 2 banana
  9. 3 grapes
  10. 4 orange

Table 最大值

table.maxn 在 Lua5.2 之后该方法已经不存在了,我们定义了 table_maxn 方法来实现。

以下实例演示了如何获取 table 中的最大值:

实例

  1. function table_maxn(t)
  2. local mn=nil;
  3. for k, v in pairs(t) do
  4. if(mn==nil) then
  5. mn=v
  6. end
  7. if mn < v then
  8. mn = v
  9. end
  10. end
  11. return mn
  12. end
  13. tbl = {[1] = 2, [2] = 6, [3] = 34, [26] =5}
  14. print("tbl 最大值:", table_maxn(tbl))
  15. print("tbl 长度 ", #tbl)

执行以上代码输出结果为:

  1. tbl 最大值: 34
  2. tbl 长度 3
注意:当我们获取 table 的长度的时候无论是使用 # 还是 table.getn 其都会在索引中断的地方停止计数,而导致无法正确取得 table 的长度。可以使用以下方法来代替:
  1. function table_leng(t)
  2. local leng=0
  3. for k, v in pairs(t) do
  4. leng=leng+1
  5. end
  6. return leng;
  7. end

table 去重

  1. function table.unique(t, bArray)
  2. local check = {}
  3. local n = {}
  4. local idx = 1
  5. for k, v in pairs(t) do
  6. if not check[v] then
  7. if bArray then
  8. n[idx] = v
  9. idx = idx + 1
  10. else
  11. n[k] = v
  12. end
  13. check[v] = true
  14. end
  15. end
  16. return n
  17. end

数组去重

  1. --数组去重函数
  2. function removeRepeat(a)
  3. local b = {}
  4. for k,v in ipairs(a) do
  5. if(#b == 0) then
  6. b[1]=v;
  7. else
  8. local index = 0
  9. for i=1,#b do
  10. if(v == b[i]) then
  11. break
  12. end
  13. index = index + 1
  14. end
  15. if(index == #b) then
  16. b[#b + 1] = v;
  17. end
  18. end
  19. end
  20. return b
  21. end
  22. --遍历数组输出
  23. function output(o)
  24. for k,v in ipairs(o) do
  25. print(k,v)
  26. end
  27. end
  28. --测试
  29. arr = {1,1,1,2,4,5,3,2,5,3,6}
  30. narr = removeRepeat(arr)
  31. table.sort(narr) --对数组排序
  32. output(narr)

结果:

  1. 1 1
  2. 2 2
  3. 3 3
  4. 4 4
  5. 5 5
  6. 6 6

排序支持自定义排序规则

  1. t =
  2. {
  3. [1] = {A = 5, B = 2},
  4. [2] = {A = 1, B = 3},
  5. [3] = {A = 3, B = 6},
  6. [4] = {A = 7, B = 1},
  7. [5] = {A = 2, B = 9},
  8. }
  9. table.sort(t, function(a, b) return a.A > b.A; end)

排序自定义排序规则时,大体规则是传参数 (a,b),当 return true 时,则 a 排在 b 前,否则相反。

  1. local a={2,3,5,52,6,74,4}
  2. table.sort(t, function(a, b) return a>b end)
  3. for k,v in ipairs(a) do print(v) end
  4. -- 74 52 6 5 4 3 2

table 泛型元素去重, 只要元素支持 == 比较

要写成完全的泛型,那么 v==a[i] 改成一个比较函数的指针 equal(v,a[i]):

  1. function removeRepeated(a)
  2. for k,v in pairs(a) do
  3. local count=0
  4. for j in pairs(a)do count=count+1 end
  5. for i=k+1,count do
  6. if v==a[i] then
  7. table.remove(a,i)
  8. end
  9. end
  10. end
  11. end
  12. local a={"a","d","c","g","d","w","c","a","g","s"}
  13. removeRepeated(a)
  14. for k,v in pairs(a) do
  15. print(k,v)
  16. end