利用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

这样给只读表赋一个新值,就会报错了

Lua只读表-LMLPHP

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打印的表,看这样的表找字段数据,会有股蛋蛋的忧伤。。。

Lua只读表-LMLPHP

05-20 08:46