元表其实也是一个表,只不过它以另一个表的查找表的形式存在,首先看下设置元表的两个方法
setmetatable(table,metatable): 对指定table设置元表(metatable),如果元表(metatable)中存在__metatable键值, setmetatable会失败 。返回值是一个设置了元表的对象(表)。
mytable = {} -- 普通表
mymetatable = {} -- 元表
setmetatable(mytable,mymetatable) -- 把 mymetatable 设为 mytable 的元表
getmetatable(table): 返回对象的元表(metatable)。
元表的定义:
在 Lua 的table 中我们可以访问对应的key来得到value值,但是却无法对两个 table 进行操作。
因此 Lua 提供了元表(Metatable),允许我们改变table的行为,每个行为关联了对应的元方法。
例如,使用元表我们可以定义Lua如何计算两个table的相加操作a+b。
当Lua试图对两个表进行相加时,先检查两者之一是否有元表,之后检查是否有一个叫"__add"的字段,若找到,则调用对应的值。"__add"等即时字段,其对应的值(往往是一个函数或是table)就是"元方法"。
再例如,当你通过键来访问 table 的时候,如果这个键没有值,那么Lua就会寻找该table的元表(假定有元表)中的"__index "键。
(1)第一种情况,如果__index包含一个表格,Lua会在表格中查找相应的键。
other = { foo = 3 }
table = setmetatable({}, { __index = other })
print(table.foo) --输出结果 3
因为table对应的foo键没有值,所以会查找__index对应的表,也就是other,other中存在
foo键,输出foo键对应的值,即3
(2) 第二种情况,如果__index包含一个函数的话,Lua就会调用那个函数,table和键会作为参数传递给函数。
__index 元方法查看表中元素是否存在,如果不存在,返回结果为 nil;如果存在则由 __index 返回结果。
mytable = setmetatable({key1 = "value1"}, {__index = function(mytable, key)
if key == "key2" then
return "metatablevalue"
else
return nil
end
end
})
print(mytable.key1,mytable.key2)
解析:
(1)mytable 表赋值为 {key1 = "value1"}。
(2)mytable 设置了元表,元方法为 __index。
(3)在mytable表中查找 key1,如果找到,返回该元素,找不到则继续。
(4)在mytable表中查找 key2,如果找到,返回该元素,找不到则继续。
(5)判断元表有没有__index方法,如果__index方法是一个函数,则调用该函数。
(6)元方法中查看是否传入 "key2" 键的参数(mytable.key2已设置),如果传
入 "key2" 参数返回 "metatablevalue",否则返回 mytable 对应的键值。
我们可以将以上代码简单写成:
mytable = setmetatable({key1 = "value1"}, { __index = { key2 = "metatablevalue" } })
print(mytable.key1,mytable.key2)
以上所说的"__add","__index" 字段本身对应的值就是元方法,元方法可能被赋值为一个函数 ,也可能被赋值为另一个表。
一些常见元方法:
__add(a, b) --加法
__sub(a, b) --减法
__mul(a, b) --乘法
__div(a, b) --除法
__mod(a, b) --取模
__pow(a, b) --乘幂
__unm(a) --相反数
__concat(a, b) --连接
__len(a) --长度
__eq(a, b) --相等
__lt(a, b) --小于
__le(a, b) --小于等于
__index(a, b) --索引查询
__newindex(a, b, c) --索引更新(PS:不懂的话,后面会有讲)
__call(a, ...) --执行方法调用
__tostring(a) --字符串输出
__metatable --保护元表
【__newindex元方法】
__newindex元方法与__index类似,__newindex用于更新table中的数据,而__index用于查询table中的数据。当对一个table中不存在的索引赋值时,在Lua中是按照以下步骤进行的:
- Lua解释器先判断这个table是否有元表;
- 如果有了元表,就查找元表中是否有__newindex元方法;如果没有元表,就直接添加这个索引,然后对应的赋值;
- 如果有这个__newindex元方法,Lua解释器就执行它,而不是执行赋值;
- 如果这个__newindex对应的不是一个函数,而是一个table时,Lua解释器就在这个table中执行赋值,而不是对原来的table。
mymetatable={}
mytable=setmetatable({key1="value1"},{
__newindex=mymetatable
})
print(mytable.key1) --value1
mytable.newkey="新值2"
print(mytable.newkey,mymetatable.newkey)--nil 新值2
mytable.key1="新值1"
print(mytable.key1,mymetatable.key1)--新值1 nil
【__add元方法】
--计算表的长度
function calTable(table)
local count = 0
for k,v in pairs(table) do
count=count+1
end
return count
end
--__add:两表相加
mytable=setmetatable({1,2,3},{
__add=function ( mytable1,newtable1)
for i=1, calTable(newtable1) do
table.insert(mytable1,newtable1[i])
end
return mytable1
end
})
secondTable={4,5,6}
mytable=mytable+secondTable
for k,v in pairs(mytable) do
print(k,v)
end
【__call元方法】
--__call:函数调用
mytable=setmetatable({10},{
__call=function (mytable1,newtable1)
local sum =0
for i=1,calTable(mytable1) do
sum=sum+mytable1[i]
end
for i=1,calTable(newtable1) do
sum=sum+newtable1[i]
end
return sum
end
})
newtable={10,20,30}
print(mytable(newtable)) --输出70
【__tostring元方法】
--__tostring:用于修改表的输出行为
mytable=setmetatable({10,20,30},{
__tostring=function ( mytable )
local sum = 0
for k,v in pairs(mytable) do
sum=sum+v
end
return "表中的所有元素和为:"..sum
end
})
print(mytable) --输出60