当前位置: 首页 > article >正文

Lua元方法

       这一章主要记录介绍 元表元方法 和使用,如果对 元表 不熟悉的同学可以闪现Lua元表。

        元方法 是元表中的某些字段,定义了表在特定操作下的行为。例如,当表进行加法运算时,Lua会检查元表中的__add字段,并使用它定义的行为。

1. 常见元方法

元方法作用
__index在表中访问不存在的键时触发。
__newindex在表中对不存在的键赋值时触发。
__add重载加法运算符 + ,如对两个表进行 时触发。
__sub重载减法运算符 - ,如对两个表进行 时触发。
__mul重载乘法运算符 * ,如对两个表进行 * 时触发。
__div重载除法运算符 / ,如对两个表进行 / 时触发。
__mod重载取模运算符 % ,如对两个表进行 % 时触发。
__pow重载幂运算符 ^ ,如对两个表进行 ^ 时触发。
__concat重载字符串连接运算符 .. ,如对两个表进行 .. 时触发。
__eq重载相等比较运算符 == ,如对两个表进行 == 时触发。
__lt重载小于比较运算符 < ,如对两个表进行 < 时触发。
__le重载小于等于比较运算符 <= ,如对两个表进行 <= 时触发。
__call将表当作函数调用时触发。
__tostring定义 tostring() 函数的输出行为。
__metatable设置元表的保护,外部无法获取或修改元表。

                                 以上包含了所有的元方法,下面会针对每一个展开记录。

2. __index

  • 功能:

    • 在访问表中不存在的键时触发。

    • 用于提供默认值,实现继承等。

  • 参数:

    • table:原始表(正在访问的表)。

    • key:访问的键名。

  • 返回值:返回键对应的值。

  • 如果 __index 元方法是一个表,则从该表中查找键的值。

  • 如果 __index 元方法是一个函数,则会调用该函数并返回结果。

  • 示例:返回默认值

local t = {}
local mt = {
    __index = function(table, key)
        return "default value for " .. key
    end
}
setmetatable(t, mt)

print(t.foo) -- 输出 "default value for foo"
  • 示例:实现简单继承

local Parent = {name = "Parent"}
local Child = {}
setmetatable(Child, {__index = Parent})

print(Child.name) -- 输出 "Parent"

3. __newindex 

  • 功能:

    • 在为表中不存在的键赋值时触发。

  • 参数:

    • table:原始表(正在访问的表)。

    • key:赋值的键名。

    • value:赋值的值。

  • 返回值:无返回值。

  • __newindex 元方法必须是一个函数,不能是表这是Lua的设计特点与 __index 明确区分。

  • 在表中赋值键时,如果元表中定义了 __newindex ,会调用该元方法不会直接修改原始表。

  • 示例:

local t = {}
local mt = {
    __newindex = function(_, key, value)
        print("Setting " .. key .. " = " .. value)
    end
}
setmetatable(t, mt)

t.foo = "bar" -- 输出 "Setting foo = bar"

4. __add 

  • 功能:

    • 定义 + 运算符的行为。

  • 参数:

    • a:第一个操作数,可以是表、数字或其他 Lua 数据类型。

    • b:第二个操作数,可以是表、数字或其他 Lua 数据类型。

  • 返回值:加法运算结果(类型任意)。

  • 当尝试对两个表使用 + 运算符时,Lua会调用元表的 __add 方法。

  • 示例:加法运算

local t1 = {value = 10}
local t2 = {value = 20}
local mt = {
    __add = function(a, b)
        return {value = a.value + b.value}
    end
}
setmetatable(t1, mt)
setmetatable(t2, mt)

local t3 = t1 + t2
print(t3.value) -- 输出 30

5. __sub

  • 功能:

    • 定义 - 运算符的行为。

  • 参数:

    • a:第一个操作数,可以是表、数字或其他 Lua 数据类型。

    • b:第二个操作数,可以是表、数字或其他 Lua 数据类型。

  • 返回值:减法运算结果(类型任意)。

  • 当尝试对两个表使用 - 运算符时,Lua会调用元表的 __sub 方法。

  • 示例:减法运算

local t1 = {value = 10}
local t2 = {value = 5}
local mt = {
    __sub = function(a, b)
        return {value = a.value - b.value}
    end
}
setmetatable(t1, mt)
setmetatable(t2, mt)

local t3 = t1 - t2
print(t3.value) -- 输出 5

 

6. __mul

  • 功能:

    • 定义 * 运算符的行为。

  • 参数:

    • a:第一个操作数,可以是表、数字或其他 Lua 数据类型。

    • b:第二个操作数,可以是表、数字或其他 Lua 数据类型。

  • 返回值:乘法运算结果(类型任意)。

  • 当尝试对两个表使用 * 运算符时,Lua会调用元表的 __mul 方法。

  • 示例:乘法运算

local t1 = {value = 4}
local t2 = {value = 5}
local mt = {
    __mul = function(a, b)
        return {value = a.value * b.value}
    end
}
setmetatable(t1, mt)
setmetatable(t2, mt)

local t3 = t1 * t2
print(t3.value) -- 输出 20

7. __div

  • 功能:

    • 定义 / 运算符的行为。

  • 参数:

    • a:第一个操作数,可以是表、数字或其他 Lua 数据类型。

    • b:第二个操作数,可以是表、数字或其他 Lua 数据类型。

  • 返回值:除法运算结果(类型任意)。

  • 当尝试对两个表使用 / 运算符时,Lua会调用元表的 __div 方法。

  • 示例:除法运算 

local t1 = {value = 10}
local t2 = {value = 2}
local mt = {
    __div = function(a, b)
        return {value = a.value / b.value}
    end
}
setmetatable(t1, mt)
setmetatable(t2, mt)

local t3 = t1 / t2
print(t3.value) -- 输出 5

8. __mod

  • 功能:

    • 定义 % 运算符的行为。

  • 参数:

    • a:第一个操作数,可以是表、数字或其他 Lua 数据类型。

    • b:第二个操作数,可以是表、数字或其他 Lua 数据类型。

  • 返回值:取模运算结果(类型任意)。

  • 当尝试对两个表使用 / 运算符时,Lua会调用元表的 __div 方法。

  • 示例:取模运算 

local t1 = {value = 10}
local t2 = {value = 3}
local mt = {
    __mod = function(a, b)
        return {value = a.value % b.value}
    end
}
setmetatable(t1, mt)
setmetatable(t2, mt)

local t3 = t1 % t2
print(t3.value) -- 输出 1

 

9. __concat

  • 功能:

    • 定义 .. 运算符的行为。

  • 参数:

    • a:第一个操作表。

    • b:第二个操作表。

  • 返回值:连接结果。

  • 当尝试对两个表使用 .. 运算符时,Lua会调用元表的 __concat 方法。

  • 示例:

local t1 = {name = "Alice"}
local t2 = {name = "Bob"}
local mt = {
    __concat = function(a, b)
        return a.name .. " & " .. b.name
    end
}
setmetatable(t1, mt)
setmetatable(t2, mt)

print(t1 .. t2) -- 输出 "Alice & Bob"

10. __call

  • 功能:

    • 定义表被当作函数调用时的行为。

  • 参数:

    • 调用时传入的参数。

  • 返回值:调用结果。

  • 当表是函数并被调用时,Lua会调用元表的 __call 方法。

  • 示例:

local t = {}
local mt = {
    __call = function(_, arg)
        return "Called with argument: " .. arg
    end
}
setmetatable(t, mt)

print(t("Hello")) -- 输出 "Called with argument: Hello"

11. __tostring

  • 功能:

    • 定义 tostring() 函数的输出行为。

  • 参数:

    • table:原始表。

  • 返回值:返回字符串表示。

  • 当表被放在 tostring() 内使用时,Lua会调用元表的 __tostring 方法。

  • 示例:

local t = {name = "Alice"}
local mt = {
    __tostring = function(table)
        return "Name: " .. table.name
    end
}
setmetatable(t, mt)

print(tostring(t)) -- 输出 "Name: Alice"

12. __metatable

  • 功能:

    • 用于保护表元,防止外部访问或修改元表。

  • 参数:

    • 无参数。

  • 返回值:如果设置了 __metatable 则返回保护值。

  • 提供外部只读访问,防止元表被修改导致意外行为。

  • 在复杂项目中,保护核心逻辑不被破坏。

  • 示例:

local t = {}
local mt = {
    __index = function(_, key)
        return "Default value for " .. key
    end,
    __metatable = "Protected"
}
setmetatable(t, mt)

print(t.foo)           -- 输出 "Default value for foo"
print(getmetatable(t)) -- 输出 "Protected"

-- 尝试修改元表
-- setmetatable(t, nil) -- 报错:cannot change protected metatable

12. 总结

元方法功能参数返回值
__index访问不存在的键table,key键的值
__newindex拦截不存在的键的赋值table,key,value
__add重载加法运算符+a,b加法结果
__concat重载字符串连接运算符..a,b连接结果
__call定义表被调用时的行为...调用结果
__tostring自定义表的字符串表示table字符串
__metatable保护元表保护值或nil
  • 通过元方法,可以深度自定义Lua表的行为,从而实现更多高级的功能。


http://www.kler.cn/a/454662.html

相关文章:

  • 我的秋招总结
  • 地理数据库Telepg面试内容整理-请描述空间索引的基本概念,如何使用它提高查询性能
  • 费舍尔信息矩阵全面讲述
  • jetson Orin nx + yolov8 TensorRT 加速量化 环境配置
  • 对gPTP上PTP安全控制的评估
  • xinput1_3.dll放在哪里?当xinput1_3.dll丢失时的应对策略:详细解决方法汇总
  • element-ui表格多级表头固定列和合并单元格
  • HTML CSS 超链
  • TCP 传输可靠性保障
  • 7-6 最小生成树-prim
  • HTTPS验证流程
  • 地理数据库Telepg面试内容整理-在Telepg数据库中,如何进行空间数据的存储与管理
  • 基于STM32的智能家居环境监控系统设计
  • 【Linux】Centos7下载npm
  • Java(三十六)集合-List ArrayList LinkedList接口
  • java基础1:处理Map
  • 《机器学习》KNN算法实现手写数字识别
  • Dots 常用操作
  • 云手机+Facebook:让科技与娱乐完美结合
  • C++--------继承
  • 了解jvm -server和-client 参数
  • 【ETCD】【实操篇(十八)】ETCD监控实战:提升系统健康与集群调试效率
  • platform_msi使用
  • 【Git】—— 使用git操作远程仓库(gitee)
  • httpclient GET 和POST 请求
  • Qt存储大整数到`JsonValue`