在Lua中有时候会遇到rawget 和 rawset 这2个函数,实际这2个函数的意思就是对表的访问和赋值 不使用元表的元方法.如下代码:
Class = {}
Class.mt = {}
function Class.New(o)
local instance = o or {}
setmetatable(instance,Class.mt)
return instance
end
Class.mt.__index = function(t, key)
return "huangyi.cc"
end
Class.mt.__newindex = function(t, key , value)
if key == "fgreen" then
rawset(t, key, value) --原始操作不使用元表方法
--t.fgreen = value --会使用元表__newindex方法
end
end
w = Class.New({})
print(rawget(w,"fgreen")) --第一次打印输出
print(w.fgreen)
w.fgreen = "nVal"
print(w.fgreen)
输出结果:

可以看到第一次打印输出是nil 说明使用rawget获取表的值时没有去访问元表的__index 方法,而是直接访问了表w ,所以输出了nil.
而第二次打印的时候没有使用rawget方法获取表的值 则调用的了元表的__index方法去获取表的值输出了元方法的返回值.
当直接对w.fgreen赋值时,使用了元方法__newindex,元方法里面又使用了rawset 设置表的值,此时不会再去调用元方法__newindex了,所以设置成功了.
如果将rawset(t, key, value)换成t.fgreen = value的话将会造成死循环,因为t.fgreen = value的直接赋值会继续去调用元方法__newindex,造成堆栈溢出.