详解Lua中的元表概念

(编辑:jimmy 日期: 2024/12/24 浏览:2)

 元表是一个表,有助于改变它连接到一个密钥集和相关的元方法的帮助下表的行为。这些元方法是强大的lua功能,如:

  •     更改/添加功能,以运算符表
  •     查看metatables当钥匙不在使用__index元表中的表可用。

有迹象表明,在处理metatables其中包括使用了两种重要的方法,

  •     setmetatable(table,metatable): 这个方法是用来设置元表的一个表。
  •     getmetatable(table): 此方法用于获取表的元表。

让我们先来看看如何设置一个表作为另一个元表。它如下所示。

复制代码 代码如下:mytable = {}
mymetatable = {}
setmetatable(mytable,mymetatable)

上面的代码可以在一个单一的行被表示为如下所示。

复制代码 代码如下:mytable = setmetatable({},{})

__index

元表的查找元表时,它不是在表中提供一个简单的例子如下所示。

复制代码 代码如下:mytable = setmetatable({key1 = "value1"}, {
  __index = function(mytable, key)
    if key == "key2" then
      return "metatablevalue"
    else
      return mytable[key]
    end
  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到元表中,如果键是没有在表中可用的,新的键的行为将被中继的方法来定义。一个简单的示例,其中元表的索引时,索引不是在主表可设定如下。

复制代码 代码如下: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

可以在上面的程序看,如果一个关键存在于主表,它只是更新它。当一个键不可用在maintable,它添加了关键metatable。

该更新用 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"

rawset 设定值,而不使用元表 __newindex。同样有rawget,获取的值,而无需使用__index。
表加入操作符的行为

一个简单的例子结合使用+运算符的两个表如下所示。

复制代码 代码如下:mytable = setmetatable({ 1, 2, 3 }, {
  __add = function(mytable, newtable)
    for i = 1, table.maxn(newtable) do
      table.insert(mytable, table.maxn(mytable)+1,newtable[i])
    end
    return 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

该__add密钥包含在元表中添加操作符+行为。表的键和相应的操作符如下所示。

详解Lua中的元表概念

 __call

完成方法调用的添加行为,使用__call声明。一个简单的例子,返回值的主表的总和与传递表。

复制代码 代码如下:mytable = setmetatable({10}, {
  __call = function(mytable, newtable)
 sum = 0
 for i = 1, table.maxn(mytable) do
  sum = sum + mytable[i]
 end
    for i = 1, table.maxn(newtable) do
  sum = sum + newtable[i]
 end
 return sum
  end
})
newtable = {10,20,30}
print(mytable(newtable))

当我们运行上面的程序,会得到下面的输出。

复制代码 代码如下:70

__tostring

更改打印语句的行为,可以用__toString元方法。一个简单的例子如下所示。

复制代码 代码如下:mytable = setmetatable({ 10, 20, 30 }, {
  __tostring = function(mytable)
    sum = 0
    for k, v in pairs(mytable) do
  sum = sum + v
 end
    return "The sum of values in the table is " .. sum
  end
})
print(mytable)

当我们运行上面的程序,会得到下面的输出。

复制代码 代码如下:The sum of values in the table is 60

如果知道元数据表的功能完全,真的可以执行很多操作,这将是不使用它非常复杂。所以尽量让工作使用metatables在元表提供不同的选择作为样本的解释,也可以创建自己的样品。

一句话新闻

一文看懂荣耀MagicBook Pro 16
荣耀猎人回归!七大亮点看懂不只是轻薄本,更是游戏本的MagicBook Pro 16.
人们对于笔记本电脑有一个固有印象:要么轻薄但性能一般,要么性能强劲但笨重臃肿。然而,今年荣耀新推出的MagicBook Pro 16刷新了人们的认知——发布会上,荣耀宣布猎人游戏本正式回归,称其继承了荣耀 HUNTER 基因,并自信地为其打出“轻薄本,更是游戏本”的口号。
众所周知,寻求轻薄本的用户普遍更看重便携性、外观造型、静谧性和打字办公等用机体验,而寻求游戏本的用户则普遍更看重硬件配置、性能释放等硬核指标。把两个看似难以相干的产品融合到一起,我们不禁对它产生了强烈的好奇:作为代表荣耀猎人游戏本的跨界新物种,它究竟做了哪些平衡以兼顾不同人群的各类需求呢?