在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,造成堆栈溢出.