Lua元表与元方法解析

本文详细介绍了Lua中的元表和元方法概念,包括元表的获取与设置,以及元方法在算术、关系、索引等操作中的应用。通过示例展示了如何使用元表实现默认值、跟踪访问和只读table等功能。
摘要由CSDN通过智能技术生成

元表

lua中的变量是没有数据类型的,值有类型。类型有八种nil,number,boolean, string, function, thread, userdata以及table。

table 作为 Lua 中唯一的数据结构,我们可以利用 table 实现面向对象编程中的类、继承、多重继承等等。

元表的概念

Lua 中的每个值都有一个元表。这个元表就是一个普通的 Lua 表,它用于定义原始值在特定操作下的行为。如果你想改变一个值在特定操作下的行为,你可以在它的元表中设置对应域。

table 和 userdata 可以有各自独立的元表,而其他类型的值则共享其类型所属的单一元表。任何 table 都可以作为任何值的元表,而一组相关的 table 也可以共享一个通用的元表。一个 table 甚至可以作为它自己的元表。

通过 getmetatable 方法可以获取一个值的元表,而 setmetatable 方法则可以设置一个值的元表。

t = {
   }
print(getmetatable(t)) --> nil
t1 = {
   }
setmetatable(t, t1)
assert(getmetatable(t) == t1)

t2 = {
   }
setmetatable(t2, t2)
assert(getmetatable(t2) == t2)

在 Lua 代码中,只能设置 table 的元表。若要设置其他类型的值的元表,则必须通过 C 代码来完成。从下面的代码也可以看出 Lua 中的所有字符串值是共用一个元表的。

print(getmetatable("hi")) --> table: 0x7fd0b14074b0
print(getmetatable("hello")) --> table: 0x7fd0b14074b0
print(getmetatable(10)) --> nil
print(getmetatable(false)) --> nil
print(getmetatable(function () end)) --> nil

setmetatable("hi", {
   }) --> error:bad argument #1 to 'setmetatable' (table expected, got string)

元方法

元方法的概念

元表中的键对应着不同的事件名;键关联的那些值被称为元方法。

例如,当你对非数字值做加操作时,Lua 会检查该值的元表中的 “__add” 域下的函数。如果能找到,Lua 则调用这个函数来完成加这个操作。
在这个例子中引用的事件为 “add” ,完成加操作的那个函数就是元方法。

你可以用 getmetatable 函数来获取任何值的元表。

使用 setmetatable 来替换一张表的元表。在 Lua 中,你不可以改变表以外其它类型的值的元表(除非你使用调试库);若想改变这些非表类型的值的元表,请使用CAPI。

表和完全用户数据有独立的元表(当然,多个表和用户数据可以共享同一个元表)。其它类型的值按类型共享元表;也就是说所有的数字都共享同一个元表,所有的字符串共享另一个元表等等。默认情况下,值是没有元表的,但字符串库在初始化的时候为字符串类型设置了元表。

元表决定了一个对象在数学运算、位运算、比较、连接、取长度、调用、索引时的行为。元表还可以定义一个函数,当表对象或用户数据对象在垃圾回收时调用它。

注意:Lua 从元表中直接获取元方法;访问元表中的元方法永远不会触发另一次元方法。

下面的代码模拟了 Lua 从一个对象 obj 中获取一个元方法的过程:

rawget(getmetatable(obj) or {
   }, "__" .. event_name)

对于一元操作符(取负、求长度、位反),元方法调用的时候,第二个参数是个哑元,其值等于第一个参数。这样处理仅仅是为了简化 Lua 的内部实现(这样处理可以让所有的操作都和二元操作一致),这个行为有可能在将来的版本中移除。(使用这个额外参数的行为都是不确定的。)

下面给出一张元表可以控制的事件的完整列表。每个操作都用对应的事件名来区分。每个事件的键名用加有 ‘__’ 前缀的字符串来表示;例如 “add” 操作的键名为字符串 “__add”。

"add": + 操作。如果任何不是数字的值(包括不能转换为数字的字符串)做加法,Lua 就会尝试调用元方法。首先、Lua 检查第一个操作数(即使它是合法的),如果这个操作数没有为 “__add” 事件定义元方法,Lua 就会接着检查第二个操作数。一旦 Lua 找到了元方法,它将把两个操作数作为参数传入元方法,元方法的结果(调整为单个值)作为这个操作的结果。如果找不到元方法,将抛出一个错误。
"sub": - 操作。行为和 “add” 操作类似。
"mul": * 操作。行为和 “add” 操作类似。
"div": / 操作。行为和 “add” 操作类似。
"mod":% 操作。行为和 “add” 操作类似。
"pow":^ (次方)操作。行为和 “add” 操作类似。
"unm":- (取负)操作。行为和 “add” 操作类似。
"idiv": // (向下取整除法)操作。行为和 “add” 操作类似。
"band"& (按位与)操作。行为和 “add” 操作类似,不同的是 Lua 会在任何一个操作数无法转换为整数时尝试取元方法。
"bor": | (按位或)操作。行为和 “band” 操作类似。
"bxor":~ (按位异或)操作。行为和 “band” 操作类似。
"bnot":~ (按位非)操作。行为和 “band” 操作类似。
"shl":<< (左移)操作。行为和 “band” 操作类似。
"shr":>> (右移)操作。行为和 “band” 操作类似。
"concat":… (连接)操作。行为和 “add” 操作类似,不同的是 Lua 在任何操作数即不是一个字符串也不是数字(数字总能转换为对应的字符串)的情况下尝试元方法。
"len":# (取长度)操作。如果对象不是字符串,Lua 会尝试它的元方法。如果有元方法,则调用它并将对象以参数形式传入,而返回值(被调整为单个)则作为结果。如果对象是一张表且没有元方法,Lua 使用表的取长度操作。其它情况,均抛出错误。
"eq":== (等于)操作。和 “add” 操作行为类似,不同的是 Lua 仅在两个值都是表或都是完全用户数据且它们不是同一个对象时才尝试元方法。调用的结果总会被转换为布尔量。
"lt":< (小于)操作。和 “add” 操作行为类似,不同的是 Lua 仅在两个值不全为整数也不全为字符串时才尝试元方法。调用的结果总会被转换为布尔量。
"le":<= (小于等于)操作。和其它操作不同,小于等于操作可能用到两个不同的事件。首先,像 “lt” 操作的行为那样,Lua 在两个操作数中查找 “__le” 元方法。如果一个元方法都找不到,就会再次查找 “__lt” 事件,它会假设 a <= b 等价于 not (b < a)。而其它比较操作符类似,其结果会被转换为布尔量。
"index":索引 table[key]。当 table 不是表或是表 table 中不存在key 这个键时,这个事件被触发。此时,会读出 table 相应的元方法。尽管名字取成这样,这个事件的元方法其实可以是一个函数也可以是一张表。如果它是一个函数,则以 table 和 key 作为参数调用它。如果它是一张表,最终的结果就是以 key 取索引这张表的结果。(这个索引过程是走常规的流程,而不是直接索引,所以这次索引有可能引发另一次元方法。)
"newindex":索引赋值 table[key] = value 。和索引事件类似,它发生在table 不是表或是表 table 中不存在key 这个键的时候。此时,会读出 table 相应的元方法。同索引过程那样,这个事件的元方法即可以是函数,也可以是一张表。如果是一个函数,则以 table、 key、以及 value 为参数传入。如果是一张表,Lua 对这张表做索引赋值操作。(这个索引过程是走常规的流程,而不是直接索引赋值,所以这次索引赋值有可能引发另一次元方法。)一旦有了 “newindex” 元方法,Lua 就不再做最初的赋值操作。(如果有必要,在元方法内部可以调用 rawset来做赋值。)
"call":函数调用操作 func(args)。当 Lua 尝试调用一个非函数的值的时候会触发这个事件(即 func 不是一个函数)。查找 func 的元方法,如果找得到,就调用这个元方法,func 作为第一个参数传入,原来调用的参数(args)后依次排在后面。

算术类的元方法:__add(加法)、__mul(乘法)、__sub(减法)、__div(除法)、__unm(相反数)、__mod(取模)、__pow(乘幂)。

关系类的元方法:__eq(等于)、__lt(小于)、__le(小于等于)。其他的关系操作符则没有单独的元方法,Lua 会将 a ~= b 转换为 not a == b ,将 a > b 转换为 a < b ,将 a >= b 转换为 a <= b 。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值