详解Lua中的元表概念

元表是一个表,有助于改变它连接到一个密钥集和相关的元方法的帮助下表的行为。这些元方法是强大的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密钥包含在元表中添加操作符+行为。表的键和相应的操作符如下所示。

 __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在元表提供不同的选择作为样本的解释,也可以创建自己的样品。

(0)

相关推荐

  • Lua中的元表(metatable)、元方法(metamethod)详解

    终于到了在实际中经常要用到的内容了--元表与元方法. 在第一次看见这两样东西的时候,可能会觉得它很深奥,但其实很好理解,虽然实际上它可能真的很深奥.(小若:停!滚粗.) 1.知道为什么1 + 1 = 2吗? 为什么在Lua中,1+1会等于2呢?(小若:难道除了Lua,其他地方就不等于2了?) 为什么数字和数字相加是合法的,为什么table和table相加就会报错?大家有想过这些问题吗?   没错,规则,这一切都只是规则而已,Lua规定了数字之间可以进行加减乘除,而table之间则不可以. 这是因

  • Lua中使用元表(metatable)执行算术类元方法实例

    上一节对元表和元方法有了一个初步的认识,这篇就来举个最简单的例子吧,加深一下印象,也为后续内容做准备. 1.元方法名 Lua其实已经规定好了各种算术操作符的元方法名字,如: __add:加法 __sub:减法 __mul:乘法 __div:除法 __unm:相反数 __mod:取模 __pow:乘幂   只要在自定义元表的时候,给这些元方法名赋予新的函数就可以实现自定义操作了. 2.例子 开始举例吧,我们新建一个自定义的元表(也就是一个table变量),用来定义一些操作: 复制代码 代码如下:

  • Lua中的元表与元方法学习总结

    前言 元表对应的英文是metatable,元方法是metamethod.我们都知道,在C++中,两个类是无法直接相加的,但是,如果你重载了"+"符号,就可以进行类的加法运算.在Lua中也有这个道理,两个table类型的变量,你是无法直接进行"+"操作的,如果你定义了一个指定的函数,就可以进行了.那这篇博文就是主要讲的如何定义这个指定的函数,这个指定的函数是什么?希望对学习Lua的朋友有帮助. Lua是怎么做的? 通常,Lua中的每个值都有一套预定义的操作集合,比如数

  • 详解Lua中的元表概念

    元表是一个表,有助于改变它连接到一个密钥集和相关的元方法的帮助下表的行为.这些元方法是强大的lua功能,如: 更改/添加功能,以运算符表 查看metatables当钥匙不在使用__index元表中的表可用. 有迹象表明,在处理metatables其中包括使用了两种重要的方法, setmetatable(table,metatable): 这个方法是用来设置元表的一个表. getmetatable(table): 此方法用于获取表的元表. 让我们先来看看如何设置一个表作为另一个元表.它如下所示.

  • 详解Lua中的数组概念知识

    数组是有序的对象的装置,它可以是包含含有多个行和列的行或多维阵列的集合的单个二维数组. 在Lua中,数组是使用索引表与整数实现的.数组的大小是不固定的,它可以增长基于我们需要受存储器限制. 一维数组 一维数组可以用一个简单的表结构来表示,可以初始化,使用一个简单的for循环读取.如下例子所示. 复制代码 代码如下: array = {"Lua", "Tutorial"} for i= 0, 2 do    print(array[i]) end 当我们运行上面的代码

  • 详解Lua中的数据类型

    Lua是动态类型语言,所以变量没有类型,仅值有类型.值可以被存储在变量中,作为参数传递,并作为结果返回. 在Lua中虽然我们没有变量的数据类型,但我们有类型的值.用于数值数据类型的列表在下面给出.  函数类型 在Lua中有一个叫做函数类型,使我们能够知道变量的类型.如下一些例子给出了下面的代码. 复制代码 代码如下: print(type("What is my type"))   --> string t=10 print(type(5.8*t))              

  • 详解Lua中的变量相关知识点

    变量不过是存储到区域项目可以操作的名称.它可以容纳不同类型的值,包括函数和表格. 变量名可以由字母,数字和下划线.它必须以字母或下划线.大写和小写字母是不同的,因为Lua是区分大小写的.有八种基本类型值在Lua中: 在Lua,尽管我们没有变量的数据类型,我们基于该变量范围的三种类型. 全局变量:所有的变量默是全局除非显式地声明为局部. 局部变量:当类型被指定为局部的一个变量,它的范围是有限的在自己的范围内使用. 表字段:这是一种特殊类型的变量,可以除了nil,包括功能不放任何东西. 在Lua变量

  • 详解Lua中的表的概念及其相关操作方法

    表格是唯一的数据结构中Lua可以帮助我们创造出不同的类型,如数组和字典. Lua使用关联数组和可不仅数字,但也有不同的零字符串索引.表格都没有固定的大小,并根据需要可以增长. Lua采用的所有陈述,包括包装的代表性表.当我们访问一个方法的字符串.格式,这意味着,我们正在访问的格式化功能的字符串封装. 表示和用法 表称为对象和它们既不值,也没有变. Lua使用构造函数表达式{}创建一个空表.它是要知道,有保存表的参考和表本身的变量之间没有固定的关系. 复制代码 代码如下: --sample tab

  • 详解Lua中if ... else语句的使用方法

    if 语句后面可以跟一个可选的else语句,当布尔表达式为假该语句执行. 语法 在Lua编程语言中的if ... else语句的语法是: 复制代码 代码如下: if(boolean_expression) then    --[ statement(s) will execute if the boolean expression is true --] else    --[ statement(s) will execute if the boolean expression is fals

  • 详解Lua中的while循环语句的使用

    在Lua编程语言中的while循环语句,只要给定的条件为真时将重复执行的目标语句. 语法 Lua编程语言中的while循环的语法是: 复制代码 代码如下: while(condition) do    statement(s) end 在这里,声明(S)可以是单一语句或语句块.该条件可以是任何表达式,并且真正是任意非零值.循环迭代当条件是true. 当条件为假,则程序控制进到紧接在循环之后的一行. 流程图: 这里,在while循环的关键点是,在循环可能不会永远运行.当条件测试结果为false,循

  • 举例详解Lua中的协同程序编程

    协同程序是协同的性质,可以把两个或更多的方法以可控制的方式执行.随着协同程序,在任何给定的时间,只有其协同程序运行之一,这在运行协同程序只能暂停其执行时,明确要求暂停. 上述定义可能看起来模糊.来告诉它更清楚,假设我们有两个方法,一个主程序方法和协同程序.当我们使用恢复功能调用协程,其开始执行,当我们调用yield功能,暂停执行.再次同协程可以继续从它被暂停的另一个恢复功能调用执行.这个过程可以继续,直到执行了协程的结束. 协同程序可用的功能 下表列出了在Lua协同程序及其相应的使用所有的可用功

  • 详解Ruby中范围的概念

    范围无处不在:1月至12月,0至9日,50至67行,依此类推. Ruby支持范围,并允许我们使用多种方式的范围: 作为序列范围 作为条件范围 作为区间范围 作为序列范围: 首先,也许是最自然的使用范围来表达序列.序列有一个起点,一个终点和序列中的连续值的方法来生产. Ruby创建'' ..''和'' ...''范围内运算符使用这些序列.这两个点的形式建立一个包容性的范围,而三个点的形式创建了一个范围,不包括指定的高值. (1..5) #==> 1, 2, 3, 4, 5 (1...5) #==>

  • 详解Ruby中的方法概念

    Ruby方法跟其他编程语言中的函数非常相似, Ruby方法用于捆绑到一个单元中的一个或多个重复的语句. 方法名称应以小写字母开始.如果一个方法的名称以大写字母开始,Ruby可能会认为这是一个常数,因此可以正确解析调用. 方法应该定义Ruby的之前调用他们,否则会引发一个异常未定义的方法调用. 语法: def method_name [( [arg [= default]]...[, * arg [, &expr ]])] expr.. end 所以,可以定义一个简单的方法如下: def meth

随机推荐