理解Lua元表的一些心得:
我们来想象一下,如果对一个table进行取值操作,但是table根本就没有这个值呢?
比如: local t = {
name = "hehe",
}
print(t.money);
输出结果当然是:nil
t只用于name这个字段,而我们却访问了它的money字段,自然是返回nil了。
但是,如果我们不希望这样呢?我们希望在访问不存在的字段时,进行一些自定义的操作呢?
没问题,Lua满足了我们,那就是,__index元方法。
在使用加法操作时,会查找__add元方法,那么,在调用table不存在的字段时,会调用__index元方法,这是一样的规则。
我们来看看代码:
local t = {
name = "hehe",
}
local mt = {
__index = function(table, key)
print("虽然你调用了我不存在的字段,不过没关系,我能探测出来:" .. key);
end
}
setmetatable(t,mt);
print(t.money);
我们给table设置了一个自定义的元表,元表的__index元方法使用了我们的函数。
运行结果如下:
[LUA-print] nil
当调用了不存在的money字段时,就会调用table元表里的__index元方法,并且会传递table和字段名两个参数。
于是,我们就可以在这个函数里做很多自定义的操作了。
2.继承的实现方法
虽然现在还没到讲解继承的时候,不过,我们可以先来稍微品尝一下。
当调用table中不存在的字段时,会调用table元表的__index元方法,这个刚刚我们已经说过了。
但是,如果这个__index元方法是一个table的话,那么,就会在这个table里查找字段,并调用。
说起来,有点混乱,看代码就清楚了:
local t = {
name = "hehe",
}
local mt = {
__index = {
money = "900,0000",
}
}
setmetatable(t,mt);
print(t.money);
留意__index,我们给它赋值了一个table,这个table中有一个money对象。
那么,当调用t的某个不存在的字段时,就会去查找__index里的table,如果找到这个字段,就调用它。
这很微妙,完全就是备胎的潜质,只有在t中找不到的时候,才会想起这个“备胎”。
输出结果如下:
3.试试继承
刚刚的例子还没法体会到“继承”的概念,我们再来看一个例子:
local smartMan = {
name = "none",
age = 25,
money = 9000000,
sayHello = function()
print("大家好,我是聪明的豪。");
end
}
local t1 = {};
local t2 = {}
local mt = {__index = smartMan}
setmetatable(t1, mt);
setmetatable(t2, mt);
print(t1.money);
t2.sayHello();
输出结果为
[LUA-print] 900,0000
[LUA-print] 大家好,我是聪明的豪。
我们定义了一个table,叫做smartMan,作为“基类”。
然后新建两个table,t1和t2,将smartMan作为元表的__index元方法。
于是,当调用t1、t2的money或者sayHello字段时,实际上就会找到smartMan的字段。
是不是很像继承的样子?