利用Lua的元表(metatable)和元函数(metafunction)可以很简单的实现此功能。
其实现大致分为三个部分
- 1.禁止在表中创建新值
- 2.禁止改变已有的值
- 3.将子表也变为只读
1.禁止在表中创建新值
使用__newindex元函数即可,它的作用就是在表赋新值时调用
local static = {
exist={exist={exist=true}},
}
setmetatable(static, {__newindex = function() assert(false, 'table is readonly\n') end})
static['not-exist'] = false
这样给只读表赋一个新值,就会报错了
2.禁止改变已有的值
这一项功能没有元函数直接提供,可以通过另一个方法巧妙的实现
原理:将传入的表作为返回表的元表__index,这样如果你想要改变值,就会转化为赋新值,这样就会调用元表的__newindex
local readonly = function(t)
local meta = {
__index = t,
__newindex = function() assert(false, 'table is readonly\n') end,
}
local locked = {}
setmetatable(locked, meta)
return locked
end local static = {
exist={exist={exist=true}},
}
static = readonly(static)
static['exist'] = false
3.将子表也变为只读
此功能使用递归很好实现,只要对子表递归调用readonly函数即可。
local static = {
exist={exist={exist=true}},
} local readonly
readonly = function(t, deep)
if deep then
for k,v in pairs(t) do
if type(v) == 'table' then
t[k] = readonly(v, deep)
end
end
end local meta = {
__index = t,
__newindex = function() assert(false, 'table is readonly\n') end,
}
local locked = {}
setmetatable(locked, meta)
return locked
end static = readonly(static, true)
static['exist']['exist']['exist'] = false -- 打印表结构,如下图
print(inspect(static))
至此,readonly表就大成了,虽然其改变了表结果不过表用起来没什么变化
当前,它也存在问题,就是打印表不方便,很不直观,对调试可能会存在影响。
下面是使用inspect打印的表,看这样的表找字段数据,会有股蛋蛋的忧伤。。。