Lua中的元表和元方法
Lua中每个值都可具有元表。 元表是普通的Lua表,定义了原始值在某些特定操作下的行为。你可通过在值的原表中设置特定的字段来改变作用于该值的操作的某些行为特征。例如,当数字值作为加法的操作数时,Lua检查其元表中的"__add"字段是否有个函数。如果有,Lua调用它执行加法。
我们称元表中的键为事件(event),称值为元方法(metamethod)。前述例子中的事件是"add",元方法是执行加法的函数。
可通过函数getmetatable查询任何值的元表。
可通过函数setmetatable替换表的元表。不能从Lua中改变其他类型的元表(除了使用调试库);必须使用C API才能做到。
表和完整的用户数据具有独立的元表(尽管多个表和用户数据可共享元表);每种其他类型的所有值共享一个元表。所以,所有数字共享一个元表,字符串也是,等等。
元表可以控制对象的数学运算、顺序比较、连接、取长、和索引操作的行为。元表也能定义用户数据被垃圾收集时调用的函数。Lua给这些操作的每一个都关联了称为事件的特定键。当Lua对某值执行其中一个操作时,检查该值是否含有元表以及相应的事件。如果有,与该键关联的值(元方法)控制Lua如何完成操作。
元表控制后面列举的操作。每个操作由相应的名字标识。每个操作的键是由其名字前缀两个下划线“__”的字符串;例如,操作“加(add)”的键是字符串"__add"。这些操作的语义通过一个Lua函数描述解释器如何执行操作作了更好的说明。
下面显示的Lua代码只是说明性的;真实的行为被硬编码到解释器中,并且比这里的模拟更加高效。这些描述中的所有函数(rawget、tonumber等等。)在§5.1中描述。特别一提,要获取给定对象的元方法,我们使用表达式
metatable(obj)[event]
它应被解读为
rawget(getmetatable(obj) or {}, event)
就是说,访问一个元方法不会调用其他元方法,而且访问没有元表的对象不会失败(只是结果为nil)。
"add": + 操作。
下面的getbinhandler函数定义Lua如何选择二元操作的处理程序。首先尝试第一操作数,如果它的类型没有定义该操作的处理程序,则尝试第二操作数。
function getbinhandler (op1, op2, event)
return metatable(op1)[event] or metatable(op2)[event]
end
通过应用该函数,op1 + op2的行为是
function add_event (op1, op2)
local o1, o2 = tonumber(op1), tonumber(op2)
if o1 and o2 then -- 两操作数都是数字
return o1 + o2 -- ‘+’此处是‘add’的原语
else -- 至少一个操作数不是数字
local h = getbinhandler(op1, op2, "__add")
if h then -- 用两个操作数调用处理程序
return (h(op1, op2))
else -- 没有可用的处理程序:缺省行为
error(...)
end
end
end
"sub": - 操作。 行为类似于“add”操作。
"mul": * 操作。 行为类似于“add”操作。
"div": / 操作。 行为类似于“add”操作。
"mod": % 操作。 行为类似于“add”操作。以o1 - floor(o1/o2)*o2为操作原语。
"pow": ^ (取幂)操作。 行为类似于“add”操作,以函数pow(来自C数学库)为操作原语。
"unm": 一元-操作。
function unm_event (op)
local o = tonumber(op)
if o then -- 操作数是数字?
return -o -- ‘-’此处是‘unm’的原语
else -- 操作数不是数字
-- 尝试由操作数取得处理程序。
local h = metatable(op).__unm
if h then-- 用操作数调用处理程序
return (h(op))
else -- 没有可用的处理程序:缺省行为
error(...)
end
end
end
"concat": .. (连接)操作。
function concat_event (op1, op2)
if (type(op1) == "string" or type(op1) == "number") and
(type(op2) == "string" or type(op2) == "number") then
return op1 .. op2 -- 字符串连接原语
else
local h = getbinhandler(op1, op2, "__concat")
if h then
return (h(op1, op2))
else
error(...)
end
end
end
"len": # 操作。
function len_event (op)
if type(op) == "string" then
return strlen(op) -- 取字符串长度原语
elseif type(op) == "table" then
return #op -- 取表长度原语
else
local h = metatable(op).__len
if h then -- 用操作数调用处理程序
return (h(op))
else -- 没有可用的处理程序:缺省行为
error(...)
end
end
end
"eq": == 操作。 函数getcomphandler定义Lua如何选择比较操作符的元方法。只有待比较的两个对象类型和选定操作对应的元方法都相同,才会选择该元方法。
function getcomphandler (op1, op2, event)
if type(op1) ~= type(op2) then return nil end
local mm1 = metatable(op1)[event]
local mm2 = metatable(op2)[event]
if mm1 == mm2 then
return mm1
else
return nil
end
end
"eq"事件定义如下:
function eq_event (op1, op2)
if type(op1) ~= type(op2) then -- 类型不同?
return false -- 对象不同
end
if op1 == op2 then -- 相等原语?
return true -- 对象相同
end -- 尝试元方法
local h = getcomphandler(op1, op2, "__eq")
if h then
return (h(op1, op2))
else
return false
end
end
a ~= b等价于not (a == b)。
"lt": < 操作。
function lt_event (op1, op2)
if type(op1) == "number" and type(op2) == "number" then
return op1 < op2 -- 数字比较
elseif type(op1) == "string" and type(op2) == "string" then
return op1 < op2 -- 词典顺序比较
else
local h = getcomphandler(op1, op2, "__lt")
if h then
return (h(op1, op2))
else
error(...);
end
end
end
a > b等价于b < a。
"le": <= 操作。
function le_event (op1, op2)
if type(op1) == "number" and type(op2) == "number" then
return op1 <= op2 -- 数字比较
elseif type(op1) == "string" and type(op2) == "string" then
return op1 <= op2 -- 词典顺序比较
else
local h = getcomphandler(op1, op2, "__le")
if h then
return (h(op1, op2))
else
h = getcomphandler(op1, op2, "__lt")
if h then
return not h(op2, op1)
else
error(...);
end
end
end
end
a >= b等价于 b <= a。注意,假定a <= b等价于not (b < a),那么当没有“le”元方法时,Lua尝试“lt”。
"index": 索引访问table[key]。
function gettable_event (table, key)
local h
if type(table) == "table" then
local v = rawget(table, key)
if v ~= nil then
return v
end
h = metatable(table).__index
if h == nil then
return nil
end
else
h = metatable(table).__index
if h == nil then
error(...);
end
end
if type(h) == "function" then
return (h(table, key)) -- 调用处理程序
else
return h[key] -- 对它重复上述操作
end
end
"newindex": 索引赋值table[key] = value。
function settable_event (table, key, value)
local h
if type(table) == "table" then
local v = rawget(table, key)
if v ~= nil then
rawset(table, key, value);
return
end
h = metatable(table).__newindex
if h == nil then
rawset(table, key, value);
return
end
else
h = metatable(table).__newindex
if h == nil then
error(...);
end
end
if type(h) == "function" then
h(table, key,value) -- 调用处理程序
else
h[key] = value -- 对它重复上述操作
end
end
"call": 当Lua调用值时被调用。
function function_event (func, ...)
if type(func) == "function" then
return func(...) -- 调用原语
else
local h = metatable(func).__call
if h then
return h(func, ...)
else
error(...)
end
end
end
相关推荐
主要介绍了举例说明Lua中元表和元方法的使用,文中--两个横线开始单行的注释,--[[加上两个[和]表示多行的注释--]],需要的朋友可以参考下
lua库说明和一些例程,lua元表和元方法,lua基本函数库,lua模式匹配
Lua的table元表自我学习笔记分享。--lua元表总结 --1、__index的运用 (调用table的一个不存在的索引时,会使用到元表的__index元方法,搜索元表是否也有改索引,__index可以是一个函数也可是一个table。)
lua学习 相关函数库和学习参考资料。 包括:lua4.0函数库 lua5.2API函数 lua-table函数库 lua捕获 lua基本函数库 lua模式匹配 lua数学库 lua文件处理 ...lua元表和元方法 string库函数 简单C访问lua
"安装羽扇如果您正在使用 ,请运行: luarocks install emoji手动的将文件夹复制到您的Lua解释器可以找到并需要它的位置: local emoji = require ( ' emoji ' )界面方法emoji.get emoji. get ( " tea " ) -> " :...
android调用lua方法
7、享元模式lua实现 行为模式 1、策略模式lua实现 2、模板方法模式lua实现 3、观察者模式lua实现 4、状态模式lua实现 5、迭代器模式lua实现 6、备忘录模式lua实现 7、命令模式lua实现 8、职责链模式lua实现 9、解释...
lua代码-lua 元表 __add方法
解析Lua实现方法 解析Lua实现方法
lua基本文档,适合初学时,包含lua函数库 lua模式匹配 lua文件处理 lua表和元方法
lua动态链接库的编译 c++调用lua的的方法 linux中调用lua的环境配置
在lua中实现类,以及继承的方法,包括了一些例子
Lua 是一个小巧的脚本语言。Lua由标准C编写而成,几乎在所有操作系统和平台上都可以编译,运行。...安装方法:安装LuaForWindows即可,不停点下一步,安装完后,桌面上会出现Lua图标和SciTE编辑器,enjoy it...
主要介绍了Lua判断一个目录或文件是否存在的方法,Lua中可以使用io.open判断文件或目录是否存在,本文总结了判断方法,并给出了一个自定义函数,需要的朋友可以参考下
详细说明:.net下实现lua方法,实现动态使用c#类库,lua语法-. net realize lua under way to realize dynamic use c# class library, lua grammar
在VC 中应用LUA的方法实例,将LUA嵌入到应用程序中,这个例子介绍的方法相对简单,只是了解一些LUA的基础知识。
2.配置项目编译环境,把lua和tolua++类库和搜索路径包括进项目来。 2.双击build_pgk.bat 会新生成lua_hello.h调用文件。 3.运行编译即可看到我们的tolua++版的Hello world.这个程序实现了脚本调用C++类的成员函数...
cpp中调用lua, 和注册方法到lua中, 及lua中调用cpp的类方法等, 库路径 -https://github.com/radiotail/eluna