一个Metatables元表是一个Table表,它通过key set 和 相关Meta方法来修改它所附加的Table行为。
处理Metatables有两种重要的方法,包括-
setmetatable(table,metatable) - 此方法用于设置表的元表。
getmetatable(table) - 此方法用于获取表的元表。
首先让无涯教程看一下如何将一个表设置为另一个表的元表。如下所示。
mytable={} mymetatable={} setmetatable(mytable,mymetatable)
上面的代码可以用单行表示,如下所示。
mytable=setmetatable({},{})
_index 方法
下面显示了一个简单的元表示例,用于在表中不可用时查找元表。
mytable = setmetatable({key1 = "value1"}, {
__index = function(mytable, key)
</span><span class="kwd">if</span><span class="pln"> key </span><span class="pun">==</span><span class="pln"> </span><span class="str">"key2"</span><span class="pln"> </span><span class="kwd">then</span><span class="pln">
</span><span class="kwd">return</span><span class="pln"> </span><span class="str">"metatablevalue"</span><span class="pln">
</span><span class="kwd">else</span><span class="pln">
</span><span class="kwd">return</span><span class="pln"> mytable</span><span class="pun">[</span><span class="pln">key</span><span class="pun">]</span><span class="pln">
</span><span class="kwd">end</span><span class="pln">
end
})
print(mytable.key1,mytable.key2)
当运行上面的程序时,将得到以下输出。
value1 metatablevalue
一起来逐步解释上面示例中发生的情况。
此处的mytable表是 {key1 =" value1"} 。
为mytable设置了元表,其中包含__index函数,将其称为元方法。
元方法可以完成查找索引" key2"的简单工作,如果找到索引,则返回" metatablevalue",否则返回对应索引的mytable值。
可以使用上述程序的简化版本,如下所示。
mytable = setmetatable({key1 = "value1"}, { __index = { key2 = "metatablevalue" } }) print(mytable.key1,mytable.key2)
__newindex 方法
当无涯教程将__newindex添加到metatable中时,如果表中没有键,则新键的行为将由meta方法定义。
mymetatable = {} mytable = setmetatable({key1 = "value1"}, { __newindex = mymetatable })print(mytable.key1)
mytable.newkey = "new value 2" print(mytable.newkey,mymetatable.newkey)
mytable.key1 = "new value 1" print(mytable.key1,mymetatable.newkey1)
当您运行上述程序时,您将获得以下输出。
value1 nil new value 2 new value 1 nil
您可以在上面的程序中看到,如果主表中存在一个键,它只会更新它。当主表中没有键时,它将把该键添加到元表中。
下面显示了另一个使用rawset函数更新同一表的示例。
mytable = setmetatable({key1 = "value1"}, {__newindex = function(mytable, key, value) rawset(mytable, key, """..value..""") end })
mytable.key1 = "new value" mytable.key2 = 4
print(mytable.key1,mytable.key2)
当运行上面的程序时,将得到以下输出。
new value "4"
__add操作符
mytable = setmetatable({ 1, 2, 3 }, {
__add = function(mytable, newtable)
</span><span class="kwd">for</span><span class="pln"> i </span><span class="pun">=</span><span class="pln"> </span><span class="lit">1</span><span class="pun">,</span><span class="pln"> table</span><span class="pun">.</span><span class="pln">maxn</span><span class="pun">(</span><span class="pln">newtable</span><span class="pun">)</span><span class="pln"> </span><span class="kwd">do</span><span class="pln">
table</span><span class="pun">.</span><span class="pln">insert</span><span class="pun">(</span><span class="pln">mytable</span><span class="pun">,</span><span class="pln"> table</span><span class="pun">.</span><span class="pln">maxn</span><span class="pun">(</span><span class="pln">mytable</span><span class="pun">)+</span><span class="lit">1</span><span class="pun">,</span><span class="pln">newtable</span><span class="pun">[</span><span class="pln">i</span><span class="pun">])</span><span class="pln">
</span><span class="kwd">end</span><span class="pln">
</span><span class="kwd">return</span><span class="pln"> mytable
end
})
secondtable = {4,5,6}
mytable = mytable + secondtable
for k,v in ipairs(mytable) do
print(k,v)
end
当运行上面的程序时,将得到以下输出。
1 1
2 2
3 3
4 4
5 5
6 6
Table内置的运算符如下所示。
| Sr.No. | Mode & Remark |
|---|---|
| 1 |
__ add 更改运算符“+”的行为。 |
| 2 |
__ sub 更改运算符"-"的行为。 |
| 3 |
__ mul 更改运算符" *"的行为。 |
| 4 |
__ div 更改运算符" /"的行为。 |
| 5 |
__ mod 更改运算符"%"的行为。 |
| 6 |
__ unm 更改运算符"-"的行为。 |
| 7 |
__ concat 更改运算符" .."的行为。 |
| 8 |
__ eq 更改运算符==的行为。 |
| 9 |
__ lt 更改运算符"<"的行为 |
| 10 |
__<= 更改运算符"<="的行为 |
__call 方法
使用__call语句完成方法调用的行为。 一个简单的示例,它返回带有传递的表的主表中的值之和。
mytable = setmetatable({10}, {
__call = function(mytable, newtable)
sum = 0
</span><span class="kwd">for</span><span class="pln"> i </span><span class="pun">=</span><span class="pln"> </span><span class="lit">1</span><span class="pun">,</span><span class="pln"> table</span><span class="pun">.</span><span class="pln">maxn</span><span class="pun">(</span><span class="pln">mytable</span><span class="pun">)</span><span class="pln"> </span><span class="kwd">do</span><span class="pln">
sum </span><span class="pun">=</span><span class="pln"> sum </span><span class="pun">+</span><span class="pln"> mytable</span><span class="pun">[</span><span class="pln">i</span><span class="pun">]</span><span class="pln">
</span><span class="kwd">end</span><span class="pln">
</span><span class="kwd">for</span><span class="pln"> i </span><span class="pun">=</span><span class="pln"> </span><span class="lit">1</span><span class="pun">,</span><span class="pln"> table</span><span class="pun">.</span><span class="pln">maxn</span><span class="pun">(</span><span class="pln">newtable</span><span class="pun">)</span><span class="pln"> </span><span class="kwd">do</span><span class="pln">
sum </span><span class="pun">=</span><span class="pln"> sum </span><span class="pun">+</span><span class="pln"> newtable</span><span class="pun">[</span><span class="pln">i</span><span class="pun">]</span><span class="pln">
</span><span class="kwd">end</span><span class="pln">
</span><span class="kwd">return</span><span class="pln"> sum
end
})
newtable = {10,20,30}
print(mytable(newtable))
当运行上面的程序时,将得到以下输出。
70
__tostring 方法
要更改print语句的行为,无涯教程可以使用__tostring元方法。一个简单的如下所示。
mytable = setmetatable({ 10, 20, 30 }, {
__tostring = function(mytable)
sum = 0
</span><span class="kwd">for</span><span class="pln"> k</span><span class="pun">,</span><span class="pln"> v </span><span class="kwd">in</span><span class="pln"> pairs</span><span class="pun">(</span><span class="pln">mytable</span><span class="pun">)</span><span class="pln"> </span><span class="kwd">do</span><span class="pln">
sum </span><span class="pun">=</span><span class="pln"> sum </span><span class="pun">+</span><span class="pln"> v
</span><span class="kwd">end</span><span class="pln">
</span><span class="kwd">return</span><span class="pln"> </span><span class="str">"The sum of values in the table is "</span><span class="pln"> </span><span class="pun">..</span><span class="pln"> sum
end
})
print(mytable)
当运行上面的程序时,将得到以下输出。
The sum of values in the table is 60