利用__index和__newindex实现默认值表、监控表、只读表

时间:2025-01-09 19:36:14

  __index和__newindex实际上相当于是在读写表的时候分别加了一道过滤的逻辑,让读写表的操作可以被监控或说回调,利用这个特性可以实现一些带有特殊功能的表。

  带有默认值的表:

setdefault = function(t, v)
setmetatable(t, {__index = function () return v end})
end s = {, , }
setdefault(s, ) -- 默认值设为0
print(s.x)

  一般访问不存在的域会返回nil,但经过上面代码处理后,访问不存在的域会返回一个默认值。为了实现公用metatable,可以将代码封装如下:

key = {}
local mt = {__index = function (t) return t[key] end}
function setdefault(t, d)
t[key] = d
setmetatable(t, mt)
end

  监控表(代理表):

t = {} --需要监控的表
local _t = t
t = {} --代理表需要为空,因为只有当读写不存在的域时,才会触发__index和__newindex local mt = {
__index = function (t, k)
print("*access to element"..tostring(k))
return _t[k]
end, __newindex = function(t, k, v)
print("*update of element " .. tostring(k) .. " to " .. tostring(v))
_t[k] = v
end
} setmetatable(t, mt) t[] = 'hello'
print(t[])

  对上述代码进行一些适当的封装,将原始表作为代理表的一个特殊的域对应的值:

local index = {}

local mt = {
__index = function (t, k)
print("*access to element " .. tostring(k))
return t[index][k]
end, __newindex = function (t, k , v)
print("*update of element " .. tostring(k) .. " to " .. tostring(v))
t[index][k] = v
end
} track = function (t)
local proxy = {}
proxy[index] = t
setmetatable(proxy, mt)
return proxy
end t = track(t)

  只读表:

function readOnly(t)
local proxy = {}
local mt = {
__index = t,
__newindex = function (t, k, v)
error("attemp to uopdaate a read-only table", )
end
} setmetatable(proxy, mt)
return proxy
end days = readOnly{"Sunday", "Monday", "Tuesday", "Wednesday","Thursday", "Friday", "Saturday"}
print(days[])
days[] = "Noday"

  上述利用__index和__newindex特性来实现的一些具有特殊表的方法都比较简单易懂,而且很好用。