--[[元方法__index:对一个表A的元素id赋值,如果在表A中元素id不存在,不对id赋值]
会调用元表中(如果元表存在)的__newindex表,如果表A中存在id这个元素,则对赋值,不调用元表中__newindex]]
local newindextable = {}
local metableA = { __newindex = newindextable }
local tableA = { id = "tableA id" }
setmetatable( tableA, metableA)
tableA.id = "change id "
tableA.age = "change age"
print(tableA.id, newindextable.id)
print(tableA.age, newindextable.age)
打印结果:
===============20180506===========
change id nil
nil change age
===============20180506===========
以上可以简写为:
local newindextable = {}
local tableA = setmetatable( { id = "tableA id" }, { __newindex = newindextable })
tableA.id = "change id "
tableA.age = "change age"
print(tableA.id, newindextable.id)
print(tableA.age, newindextable.age)
rawset/rawget:绕过metatable的行为约束,强制对原始表进行一次原始的操作;一次原始的操作其实并不会加速代码执行的速度,效率一样。
格式:
rawset(table, key, value)
rawget(table, key)
如果我们有以下需求:
- 访问时,不想从 __index 对应的元方法中查询值
- 更新时,不想执行 __newindex 对应的元方法
- 在 __newindex 元方法中,设置table的key/value时,不想陷入死循环而爆栈
那么,我们可以考虑使用raw方法。