神经刀 参考文章

lua 元方法 _newindex 实现只读的 table

根据 __newindex 的这一个特性,可以用来跟踪一个 table 赋值更新的操作,如果是一个只读的 table ,可以通过 __newindex 来实现下面是代码 lua 中 __newindex 的调用机制跟 __index 的调用机制是一样的,当访问 table 中一个不存在的 key,并对其赋值的时候,lua 解释器会查找 __newindex 元方法,如果存在,调用该方法,如果不存在,直接对原 table 索引进行赋值操作。

local t = {}
local prototype = {}
local mt = {__index = function(t,k)
    return prototype[k]
    end
,__newindex = function(t,k,v)
    print("attempt to update a table k-v")
    prototype[k] = v
end}
 
setmetatable(t,mt)
t[2] = "hello"
print(t[2])

输出:

>attempt to update a table k-v
hello

根据 __newindex 的这一个特性,可以用来跟踪一个 table 赋值更新的操作,如果是一个只读的 table ,可以通过 __newindex 来实现下面是代码:

function readOnly(t)
    local proxy = {}  --定义一个空表,访问任何索引都是不存在的,所以会调用__index 和__newindex
    local mt = {
    __index = t, ---__index 可以是函数,也可以是table,是table的话,调用直接返回table的索引值
    __newindex = function(t,k,v)
        error("attempt to update a read-only table",2)
    end
    }
    setmetatable(proxy,mt)
    return proxy
end
 
days = readOnly{"Sunday","Monday","Tuesday","Wednessday","Thursday","Friday","Saturday"} 
print(days[1])
days[2] = "hello"  --这一行就非法访问了

输出:

>Sunday
>: attempt to update a read-only table