无涯教程-Lua - Metatables(元表)

97 阅读3分钟

一个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

参考链接

www.learnfk.com/lua/lua-met…