前言
记得原来我们总结过一个查询元表的函数getmetatable(), 但是有查询必然会有设置,今天我们就一起来学习一下有关元表的设置的操作,以及设置元表时需要注意的相关问题,根据我的理解,Lua中模拟类以及类的继承关系也是通过元表的方式实现的,今后我们会继续深入学习。
内容
setmetatable
- setmetatable(table, metatable)
- 解释:Lua中的这个函数只能设置所给表的元表,如果要想改变其他类型的元表只能在C语言中实现,不能通过lua函数设置,如果第二个参数
matetable
为nil
,则将table的元表设置为空。如果原来table的元表中包含一个"__metatable"
字段,那么函数将会抛出一个错误。
usage
- 首先我们新建一个文件将文件命名为setmetatabletest.lua然后编写代码如下:
-- 定义一个数字
local aNumber = 100
-- 定义一个普通的元表
local tab1 = {
x = 13,
y = 36,
}
-- 设置数字的元表,将会如下错误,为了不影响下面运行,我们就注释掉了
-- 错误提示:bad argument #1 to 'setmetatable' (table expected, got number)
-- setmetatable(aNumber, tab1)
-- 没有元表时打印变量
print("\ntab1.z =", tab1.z)
-- 设置元表后打印变量
local m1 = { z = 36}
setmetatable(tab1, {__index = m1})
print("\ntab1.z =", tab1.z)
-- 设置元表为空后打印变量
setmetatable(tab1, nil)
print("\ntab1.z =", tab1.z)
-- 设置为有字段"__metatable"的元表
setmetatable(tab1, {__index = m1, __metatable = 'I have metatable'})
print("\ntab1.z =", tab1.z)
-- 再次设置元表会报错,因为原来的元表有'__metatable',起到保护作用
setmetatable(tab1, {__index = m1})
- 运行结果
总结
- 注释中的代码运行会报错,提示信息显示函数的第一个参数只能是'table'类型,也就表示再Lua中不能给其他类型设置元表。
- 设置元表后,如果元表中包含'__index'字段,那么原来table中不包含的字段就可以到元表中查找,如果元表中存在就可以拿过来用。
- 当然你可以在不需要元表的时候删除它,方法就是将第二个参数改为
nil
,将元表设置为空。 - 有一种特殊的情况就是当元表中包含'__metatable'的时候,该元表将不能被修改,最后的错误也验证了这一点。