本文章转自菜鸟教程
元表:关键字Metatable
setmetatable (table.metatable):对指定table设置元表(metatable),如果元表(metatable)中存在__metatable键值,setmetatable会失败
getmetatable(table):返回对象的元表(metatable)。
代码片段:
mytable = {} -- 普通表
mymetatable = {} -- 元表
setmetatable(mytable,mymetatable) -- 把 mymetatable 设为 mytable 的元表
以上代码也可以直接写成一行:
mytable = setmetatable ( { } , { } )
以下为返回对象元表:
getmetatable (mytable) -- 这个函数会返回普通表的元表(mymetatable)
元方法:关键字__index
这是metatable最常用的键。
当你通过__index来访问table的时候,如果这个键没有值,那么Lua就会寻找该table的metatable( 假定有metatable )中的__index键。如果__index包含一个表格,Lua会在表格中查找相应的键。
如果__index包含一个函数的话,Lua就会调用那个函数,table和键会作为参数传递给函数。__index元方法查看表中元素是否存在,如果不存在,返回结果为nil;如果存在则由__index返回结果。
代码片段:
mytable = { key1 = 123}
mymetatable = {
__index = function()
return 456
end
}
mytable = setmetatable( mytable, mymetatable )
print(mytable.key1)
print(mytable.key2)
总结
Lua查找一个表元素时的规则,其实就是如下3个步骤:
- 1.在表中查找,如果找到,返回该元素,找不到则继续
- 2.判断该表是否有元表,如果没有元表,返回nil,有元表则继续。
- 3.判断元表有没有__index方法,如果__index方法为nil,则返回nil;如果__index方法是一个表,则重复1、2、3;如果__index方法是一个函数,则返回该函数的返回值。
__newindex 元方法:
__newindex 元方法用来对表更新,__index则用来对表访问。
当你给表的一个缺少的索引复制,解释器就会查找__newindex 元方法:如果存在则调用这个函数而不进行赋值操作。
以下实例演示了__newindex 元方法的应用:
mymetatable = { }
mytable = setmetatable( { key1 = "value1" },{ __newindex = mymetatable } )
print (mytable.key1)
mytable.newkey = "新值2"
print(mytable.newkey,mymetatable.newkey)
mytable.key1 = "新值1"
print (mytable.key1,mymetatable.key1)