“add” 元方法 (metamethod),来计算两个 Lua 数组的并集;或者重载 “index” 方法,来定义我们自己的 Hash 函数。Lua 提供了两个十分重要的用来处理元表的方法,如下:
- setmetatable(table, metatable):此方法用于为一个表设置元表。
- getmetatable(table):此方法用于获取表的元表对象。
__index
当你通过键来访问 table 的时候,如果这个键没有值,那么Lua就会寻找该table的metatable(假定有metatable)中的index 键。如果index包含一个表格,Lua会在表格中查找相应的键。
tab = { "a", "b", "c" }
tab = setmetatable(tab, { __index = function(tab, key)
if key >= 4 then
return "d"
end
end })
print(tab[3])
print(tab[4])
打印结果
c
d
Lua查找一个表元素时的规则,其实就是如下3个步骤:
- 1.在表中查找,如果找到,返回该元素,找不到则继续
- 2.判断该表是否有元表,如果没有元表,返回nil,有元表则继续。
- 3.判断元表有没有index方法,如果index方法为nil,则返回nil;如果index方法是一个表,则重复1、2、3;如果index方法是一个函数,则返回该函数的返回值。
__tostring
__tostring可以修改table转化为字符串
local version = { major = 1, minor = 2, patch = 0 }
print(version)
version = setmetatable(version, { __tostring = function(t)
return string.format("%d.%d.%d", t.major, t.minor, t.patch)
end })
print(version)
输出结果如下所示,如果没有设置元方法 只会输出这个 table 的地址。用 setmetatable ,重新定义了 version 的 __tostring 方法,就可以打印出版本号: 1.2.0
table: 0x06719320
1.2.0
和 __index 元函数的结合使用
local version = { major = 1 }
version = setmetatable(version, {
__tostring = function(t)
return string.format("%d.%d.%d", t.major, t.minor, t.patch)
end,
__index = function(t,key)
return 0
end
})
print(version)
输出结果
1.0.0
__call
__call可以让table当做一个函数来使用
local _M = {}
_M.__call = function(t,...)
for _, v in ipairs { ... } do
print(v)
end
end
t = {}
setmetatable(t, _M)
t(1, 2, 3)
__newindex
local mt = {}
--第一个参数是表自己,第二个参数是索引,第三个参数是赋的值
mt.__newindex = function(t, index, value)
string.format(index)
print(index, value)
end
t = { key = "it is key" }
setmetatable(t, mt)
--为表中不存在的newKey索引赋值,调用了元表的__newIndex元方法,输出了参数信息
t.newKey = "it is new key"
print(t.newKey)
输出结果,表中的newKey索引值还是空,上面看着是一个赋值操作,其实只是调用了__newIndex元方法,并没有对t中的元素进行改动
newKey it is new key
nil