
手册中的相关条目是Section 2.8.
一个很好的教程可以在here或here找到.
Metatable只是一个像其他任何表一样的表,但是在另一个表上设置为Metatable(我将进一步称之为基表,以区分两个表).
Metatable可以包含任何内容,但特殊键(以双下划线开头)是有趣的.在此表中设置为此键的值将在特殊情况下调用.哪个场合取决于哪个键.最有趣的是:
> __index:只要查找基表中的键但不存在,就会使用它.这可以包含表,其中将查找键,或者一个函数,它将传递原始表和键.这可以用于实现表上的方法(OOP样式),重定向,通过案例,设置默认值等
> __newindex:只要在表中分配新密钥(之前为零),就会使用它.如果是表,则将在该表中分配密钥.如果它是一个函数,那么该函数将传递原始表,键和值.这可用于控制对表的访问,预处理数据,分配的重定向.
> __call:如果您使用eg,则可以设置要调用的函数.表().
> __add,__ sub,__ mul,__ div,__ mod用于实现二进制 *** 作,
> __unm用于实现一元 *** 作,
> __concat用于实现连接(..运算符)
> __len用于实现长度运算符(#)
> __eq,__ lt,__ le用于实现比较
使用__index&时需要了解一件小事. co.:在这些方法中,你应该使用rawget和rawset以防止每次再调用Metamethod,从而导致循环.
举个例子:
t={1,2,3} -- basetablemt={} -- Metatablemt.__index=function(t,k) print("__index event from "..tostring(t).." key "..k) return "currently unavailable"endmt.__newindex=function(t,k,v) print("__newindex event from "..tostring(t).." key: "..k.." value: "..v) if type(k)=="string" then rawset(t,v:reverse()) else rawset(t,v) endendmt.__call=function(t,...) print("call to table "..tostring(t).." with arguments: ".. table.concat({...},',')) print("All elements of the table:") for k,v in pairs(t) do print(k,v) endendsetMetatable(t,mt)t[4]="foo" -- this will run the __newindex methodprint(t[5]) -- this will run the __index methodt("foo","bar")-- Multiple fall through example:t={}mt={}mt2={}setMetatable(t,mt) -- Metatable on base tablesetMetatable(mt,mt2) -- second layer of Metatablemt.__index=function(t,k) print('key '..k..' not found in '..namelookup[t]) return getMetatable(t)[k] end -- trIEs looking nonexistant indexes up in mt.mt2.__index=mt.__index -- function was written portably,reuse it.t[1]='A'mt[2]='B'mt2[3]='C'namelookup={[t]="t",[mt]="mt",[mt2]="mt2"}print(t[1],t[2],t[3],t[4]) 现在这些只是愚蠢的例子,你可以做更复杂的事情.看看这些例子,看看Programming in Lua的相关章节,并进行实验.并尽量不要混淆;)
总结以上是内存溢出为你收集整理的Lua中表和Metatables之间的差异全部内容,希望文章能够帮你解决Lua中表和Metatables之间的差异所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)