add” 元方法 (metamethod),来计算两个 Lua 数组的并集;或者重载 “index” 方法,来定义我们自己的 Hash 函数。Lua 提供了两个十分重要的用来处理元表的方法,如下:

  • setmetatable(table, metatable):此方法用于为一个表设置元表。
  • getmetatable(table):此方法用于获取表的元表对象。

__index

当你通过键来访问 table 的时候,如果这个键没有值,那么Lua就会寻找该table的metatable(假定有metatable)中的index 键。如果index包含一个表格,Lua会在表格中查找相应的键。

  1. tab = { "a", "b", "c" }
  2. tab = setmetatable(tab, { __index = function(tab, key)
  3. if key >= 4 then
  4. return "d"
  5. end
  6. end })
  7. print(tab[3])
  8. print(tab[4])

打印结果

  1. c
  2. d

Lua查找一个表元素时的规则,其实就是如下3个步骤:

  • 1.在表中查找,如果找到,返回该元素,找不到则继续
  • 2.判断该表是否有元表,如果没有元表,返回nil,有元表则继续。
  • 3.判断元表有没有index方法,如果index方法为nil,则返回nil;如果index方法是一个表,则重复1、2、3;如果index方法是一个函数,则返回该函数的返回值。

__tostring

__tostring可以修改table转化为字符串

  1. local version = { major = 1, minor = 2, patch = 0 }
  2. print(version)
  3. version = setmetatable(version, { __tostring = function(t)
  4. return string.format("%d.%d.%d", t.major, t.minor, t.patch)
  5. end })
  6. print(version)

输出结果如下所示,如果没有设置元方法 只会输出这个 table 的地址。用 setmetatable ,重新定义了 version 的 __tostring 方法,就可以打印出版本号: 1.2.0

  1. table: 0x06719320
  2. 1.2.0

和 __index 元函数的结合使用

  1. local version = { major = 1 }
  2. version = setmetatable(version, {
  3. __tostring = function(t)
  4. return string.format("%d.%d.%d", t.major, t.minor, t.patch)
  5. end,
  6. __index = function(t,key)
  7. return 0
  8. end
  9. })
  10. print(version)

输出结果

  1. 1.0.0

__call

__call可以让table当做一个函数来使用

  1. local _M = {}
  2. _M.__call = function(t,...)
  3. for _, v in ipairs { ... } do
  4. print(v)
  5. end
  6. end
  7. t = {}
  8. setmetatable(t, _M)
  9. t(1, 2, 3)

__newindex

  1. local mt = {}
  2. --第一个参数是表自己,第二个参数是索引,第三个参数是赋的值
  3. mt.__newindex = function(t, index, value)
  4. string.format(index)
  5. print(index, value)
  6. end
  7. t = { key = "it is key" }
  8. setmetatable(t, mt)
  9. --为表中不存在的newKey索引赋值,调用了元表的__newIndex元方法,输出了参数信息
  10. t.newKey = "it is new key"
  11. print(t.newKey)

输出结果,表中的newKey索引值还是空,上面看着是一个赋值操作,其实只是调用了__newIndex元方法,并没有对t中的元素进行改动

  1. newKey it is new key
  2. nil