Lua元表的理解(二) __index __newindex 的实践

本文深入探讨Lua元表的使用,包括如何通过__index设置table默认值,如何记录table的访问操作,以及利用元表实现只读table。通过代理空表,可以有效地监控和控制table的行为。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、具有默认值的table --通过__index实现

首先 我们都知道table中的任何字段的默认值是nil  我们可以通过元表来修改table的默认值

function setDefault(t, d)
	local mt = { __index = function() return d end}
	setmetatable(t, mt)
end

function main()
	local table1 = {x = 100, y = 1000}	
	print(table1.x, table1.y,table1.q) -- 100 1000 nil
	setDefault(table1, 88888)
	print(table1.x, table1.y, table1.q) --100 1000 88888
	
end

上面的代码中 setDefault函数为所有需要默认值的table创建了一个新的元表 这样开销太大 并且元表中默认值是与元方法关联在一起的 所以setDefault函数无法为所有的table使用同一个元表 如何解决不同默认值使用同一个元表呢 只要增加一个字段即可

--利用___这个字段将每个table的默认值保存在它本身中
local mt = { __index = function(t) return t.___ end}
function setDefault1(t, d)
	t.___ = d
	setmetatable(t, mt)
end

__是为了防止名字冲突而起的名字 如果还担心名字冲突 可以再增加一个key 唯一标识

--防止名字冲突 可以用本身table中的一个key保存默认值
local key = {}
function setDefault3(t, d)
	local mt = { __index = function(t) return t[key] end}
	t[key] = d
	setmetatable(t, mt)
end

function main()
	local table1 = {x = 100, y = 1000}	
	print(table1.x, table1.y,table1.q)
	setDefault3(table1, 88888)
	print(table1.x, table1.y, table1.q)
	
	local table2 = {name = "zhangyin", money = 10000000}
	print(table2.name, table2.money, table2.height)
	setDefault3(table2, 155)
	print(table2.name, table2.money, table2.height)
	
end

main()

只需要创建一个table 并用它作为key即可 每一个新建的table都是一个唯一的地址

(二)记录table的访问

__index和__newindex都是在table中没有访问的index时才发挥作用的  只有将一个table保持为空  才有可能捕捉到所有对他的访问 解决方法:为了监视一个table的所有访问 就应该为真正的table创建一个代理 这个代理就是一个空的table 其中__index  __newindex元方法可以用来跟踪所有的访问 并将访问重定向到原来的table上

local t = { x = 10000, y = "zzzzzzzzzzz"} 以下是修改x的值 以及xiugait中不存在字段z的值 对这些操作的跟踪

--原来的表 可能在别处创建
local t = { x = 10000, y = "zzzzzzzzzzz"}
print("before opreate table element")
for key, value in pairs(t) do
	print(key, value)
end
--保持对原table的一个引用
local _t = t

--创建tt的代理表 空表
rt = {}

--创建元表
local mt = {
__index = function(rt, key)
	print("access to element " .. tostring(key))
	return _t[key] --访问原来的table _t是原来的t的一个引用
end,
__newindex = function(rt, key, value)
print("update of element " .. tostring(key) .. " to " .. tostring(value))
_t[key] = value  --更新原表中的元素
end
}

setmetatable(rt, mt) 

rt.x = 10  --update of element x to 10
print("x = " .. rt.x) --access to element  x = 10

rt.z = 888888888888 --元表中不存在字段
print("z = " .. rt.z)
print("------------------------------")
print("after opreate table element")
for key, value in pairs(t) do
	print(key, value)
end

再次强调为什么需要空表跟踪table的原因?

当我们访问一个table的元素时,第一步在table中查找是否存在该元素 如果存在则直接返回 如果不存在 第二步判断该table是否有元表 如果没有元表 则直接返回nil  如果有元表 则第三步 查看是否存在__index元方法 如果不存在直接返回nil 如果有在根据__index查找元素

当我们修改一个table的元素时,第一步先查找是否存在要修改的元素 如果存在则直接修改 如果没有 则第二步判断该表是否具有元表 如果没有元表 则直接会在一开始的表中创建对应的字段 如果存在元表 并且有设置__newindex 那么会进行相应的处理

如果未设置__newindex  会在一开始的表中创建对应的字段

从上面的描述可以看出 __index  和 __newindex 是在table中没有所需访问的index时才发挥作用的 那么如果我们想要个跟踪table的操作行为 需要一张空表 每次对这个空表操作的时候 就会使用__index 或者__newindex这些元方法 在元方法中对原始table进行访问和操作 并打印跟踪信息 而之前创建的那个空表 就是代理

(三)只读table

通过代理 只要在所有想要有对table元素有更新操作的时候引发一个错误即可

--[[
只读table
--]]

function readonly(t)
	local rt = {}
	local mt = {
				__index = t,
				__newindex = function(t, key, value)
					error("attemp to update a readonly table", 2)
				end
	}
	setmetatable(rt, mt)
	return rt
end

function  main()
	local t = { x = 10000, y = "zzzzzzzzzzz"}
	local rt = readonly(t)
	
	rt.x = 300 --attemp to update a readonly table
end
main()

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值