Lua | 每日一练 (2)

💢欢迎来到张胤尘的技术站

💥技术如江河,汇聚众志成。代码似星辰,照亮行征程。开源精神长,传承永不忘。携手共前行,未来更辉煌💥

文章目录

Lua | 每日一练 (2)

题目

简述 lua 中的元表(metatable),它的作用是什么?有哪些使用场景?

参考答案

lua 中元表主要作用改变表的默认行为(函数)。通过使用元表,可以为表定义特殊的元方法,从而使用运算符重载、访问控制、面向对象等功能。

  • 自定义运算符
lua 复制代码
local v1 = { x = 3, y = 4 } -- 定义表 v1,其中包含元素 x 和 y
local v2 = { x = 1, y = 2 }

-- 定义自定义运算函数
local add = function(a, b)
    return { x = a.x + b.x, y = a.y + b.y }
end

-- 设置 v1 表的元表,并设置 add 元方法
setmetatable(v1, { __add = add })

-- 设置 v2 表的元表,并设置 add 元方法
setmetatable(v2, { __add = add })

local ret = v1 + v2
print(ret.x, ret.y) -- 4	6
  • 控制属性访问

通过 __newindex__index 元方法,可以实现只读属性或者动态字段

lua 复制代码
local readonly = { x = 10 }
local proxy = {}	-- 添加代理表

local mt = {
    -- 读取操作:从原始表中获取值
    __index = function(tb, key)
        return readonly[key]
    end,
    
    -- 写入操作:拦截所有赋值操作并抛出错误
    __newindex = function(tb, key, value)
        error("attempt to modify a read-only table")
    end
}

setmetatable(proxy, mt)	-- 对代理表设置元表

print(proxy.x) -- 10
proxy.x = 20   -- attempt to modify a read-only table
proxy.y = 1    -- attempt to modify a read-only table
  • 模拟面向对象编程
lua 复制代码
local Animal = {}

function Animal:new(name)
    local obj = {}
    setmetatable(obj, self)
    self.__index = self
    obj.name = name or ""
    return obj
end

function Animal:speak()
    print(self.name .. " makes a sound.")
end

-- 创建对象
local dog = Animal:new("dog")
dog:speak() -- dog makes a sound.
  • 数据验证
lua 复制代码
local function validate(tbl)
    local proxy = {}	-- 添加代理表
    local mt = {	
        __index = tbl,
        __newindex = function(t, k, v)
            if type(v) ~= "number" then
                error("only numbers are allowed")
            end
            if v < 0 then
                error("negative values are not allowed")
            end
        end
    }
    setmetatable(proxy, mt)	-- 代理表设置元表
    return proxy
end

local validated_table = validate({ a = 1, b = 2 })
validated_table.a = -1       -- negative values are not allowed
validated_table.b = "string" -- only numbers are allowed

🌺🌺🌺撒花!

如果本文对你有帮助,就点关注或者留个👍

如果您有任何技术问题或者需要更多其他的内容,请随时向我提问。

相关推荐
Ruihong13 小时前
Vue withDefaults 转 React:VuReact 怎么处理?
vue.js·react.js·面试
kyriewen14 小时前
别再这样写 async/await 了:我在 Code Review 中见过最多的 8 个错误
前端·javascript·面试
烬羽19 小时前
字符串算法入门:从反转字符串到回文判断,面试不再慌
算法·面试
云技纵横19 小时前
一个 @Async,把 @Transactional 的事务边界打穿了
后端·面试
想要成为糕糕手20 小时前
Harness Engineering:大模型时代的“马鞍”——从记忆层开始,让AI真正为你所用
面试·ai编程·claude
kyriewen1 天前
我手写了一个 EventEmitter,面试官追问了 6 个问题——第 4 个我没答上来
前端·javascript·面试
她的男孩1 天前
后台接口加密别只会 HTTPS,ForgeAdmin 的 RSA + SM4/AES 源码拆解
后端·面试·开源
Randyliu2 天前
20260508-Agent搭建记录以及对ReAct框架的理解
面试·agent
ZzT2 天前
公司用 AI 筛简历,他写了个 AI 帮你挑公司
面试·aigc·ai编程
PBitW2 天前
GPT训练我的第四天,被打惨了!!!😭😭😭
前端·javascript·面试