“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 thenreturn "d"endend })print(tab[3])print(tab[4])
打印结果
cd
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: 0x067193201.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 0end})print(version)
输出结果
1.0.0
__call
__call可以让table当做一个函数来使用
local _M = {}_M.__call = function(t,...)for _, v in ipairs { ... } doprint(v)endendt = {}setmetatable(t, _M)t(1, 2, 3)
__newindex
local mt = {}--第一个参数是表自己,第二个参数是索引,第三个参数是赋的值mt.__newindex = function(t, index, value)string.format(index)print(index, value)endt = { 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 keynil
