好吧,我写文章的进度已经赶不上看书的进度了,简单的几段文字就够我唠叨一篇文章了。
今天继续来说说元方法,与__index有点相似的__newindex元方法。
1.查询与更新
上一篇文章我们介绍了__index元方法,总结来说,__index元方法是用于处理调用table中不存在的字段。
注意,【调用】这个词,只是调用,而不是赋值。
如果,我们要对table中某个不存在的字段赋值呢?(小若:就,直接赋值啊!)
没错,我们直接就能赋值了,不会报错的。
问题是,如果我想监控这个操作呢?如果有人想对table不存在的字段进行赋值的时候,我想进行一些额外的处理呢?
这时候就要用到__newindex。
大家要记住这句话:__index用于查询,__newindex用于更新。
等会不要混乱了, 初次接触的话,有可能会混乱。
2.看看普通的赋值情况
我们先来看看正常情况下的赋值,如代码:
local smartMan = { name = "none", money = 9000000, sayHello = function() print("大家好,我是聪明的豪。"); end } local t1 = {}; local mt = { __index = smartMan, } setmetatable(t1, mt); t1.sayHello = function() print("en"); end; t1.sayHello();
这是上一篇用过的例子,一个模仿继承结构的例子。
来分析一下,mt作为t1的元表,设置__index为smartMan。
于是,当我们调用t1中不存在的字段时,就会自动去smartMan中查找。
比如我们调用了t1.sayHello(),自然能找到对应的函数。
先来看看输出结果:
[LUA-print] en
但这不影响,给这个字段赋值,然后再调用t1.sayHello(),发现是成功的。
这和我们以往的做法一样,对table做正常的赋值操作,不管table本身是否存在这个字段。
3.监控赋值
好了,普通情况我们已经试过了,如果我们想监控table的赋值操作呢?
对于不存在的字段,我们不需要被赋值呢?想要制作一个只读的table呢?
如果你有这些想法,那么欢迎拨打屏幕下方的号码,前10位打进的还赠送价值..(小若:停!)
那么,如果你有这些想法,请看看下面的代码:
local smartMan = { name = "none", money = 9000000, sayHello = function() print("大家好,我是聪明的豪。"); end } local t1 = {}; local mt = { __index = smartMan, __newindex = function(table, key, value) print(key .. "字段是不存在的,不要试图给它赋值!"); end } setmetatable(t1, mt); t1.sayHello = function() print("en"); end; t1.sayHello();
留意mt元表,我们给它加了一个__newindex。
运行代码,输出结果如下:
[LUA-print] sayHello字段是不存在的,不要试图给它赋值! [LUA-print] 大家好,我是聪明的豪。
(小若:为什么?sayHello字段不是存在的么?为什么会说不存在呢?)
这里有一个地方要注意的,t1中确实是不存在sayHello字段的,它只是因为有元表存在,而元表里的__index元方法的值是smartMan这个table。
从而,可以在t1找不到sayHello字段的时候,去smartMan中寻找。
但,实际上,t1确实是不存在sayHello字段的,不知道大家能绕明白不?
因此,当试图给t1的sayHello字段赋值时,Lua判定sayHello字段是不存在的,所以会去调用元表里的__newindex元方法。
__newindex元方法被调用的时候会传入3个参数:table本身、字段名、想要赋予的值。
4.隔山打牛,通过给一个table给另一个table的字段赋值
和__index一样,__newindex元方法也可以赋予一个table值。
这种情况下就有点意思了,先看看代码:
local smartMan = { name = "none", } local other = { name = "大家好,我是很无辜的table" } local t1 = {}; local mt = { __index = smartMan, __newindex = other } setmetatable(t1, mt); print("other的名字,赋值前:" .. other.name); t1.name = "小偷"; print("other的名字,赋值后:" .. other.name); print("t1的名字:" .. t1.name);
这次的代码和刚刚差不多,但是我们新加了一个other的table,然后把other作为__newindex的值。
于是,当给t1的name字段赋值时,就会发生一些奇怪的事情…
先来看看输出结果:
[LUA-print] other的名字,赋值前:大家好,我是很无辜的table [LUA-print] other的名字,赋值后:小偷 [LUA-print] t1的名字:none
(实际上t1的name字段还是不存在的,它只是通过__index找到了smartMan的name字段,这个就不唠叨了。)
于是,我们给t1的name赋值的时候,实际上是给other的name赋值了。
好吧,可怜的other。
5.总结规则
这就是__newindex的规则:
a.如果__newindex是一个函数,则在给table不存在的字段赋值时,会调用这个函数。
b.如果__newindex是一个table,则在给table不存在的字段赋值时,会直接给__newindex的table赋值。
6.结束
好了,关于元表和元方法的基础内容基本上告一段落了,接下来还有一篇关于元表和元方法的文章,也是一些比较零散的知识点。
之后,还会提到元表和元方法的,因为它们实在是太重要了。
本文向大家介绍理解Lua中的__index和__newindex,包括了理解Lua中的__index和__newindex的使用技巧和注意事项,需要的朋友参考一下 总结: 如果在元table中去找相应的操作,例如__index,__newindex等,如果有则直接访问,如果没有就新添加进元table中
本文向大家介绍Lua中强大的元方法__index详解,包括了Lua中强大的元方法__index详解的使用技巧和注意事项,需要的朋友参考一下 今天要来介绍比较好玩的内容——__index元方法 1.我是备胎,记得回头看看 咳咳,相信每一位女生都拥有或者不知不觉中拥有了一些备胎,啊,当然,又或许是成为过别人的备胎。 没有备胎的人,就不是完整的人生。(小若:停!) 我们来想象一下,如果对一个tabl
本文向大家介绍Lua中的__index方法详解,包括了Lua中的__index方法详解的使用技巧和注意事项,需要的朋友参考一下 当我们访问一个表的不存在的域,返回结果为nil,这是正确的,但并不一定正确。实际上,这种访问触发lua解释器去查找__index metamethod:如果不存在,返回结果为nil;如果存在则由__index metamethod返回结果。 这个例子的原型是一种继承。假设
本文向大家介绍详解Lua中的元表概念,包括了详解Lua中的元表概念的使用技巧和注意事项,需要的朋友参考一下 元表是一个表,有助于改变它连接到一个密钥集和相关的元方法的帮助下表的行为。这些元方法是强大的lua功能,如: 更改/添加功能,以运算符表 查看metatables当钥匙不在使用__index元表中的表可用。 有迹象表明,在处理metatables其中包括使用了两种重要的方法
本文向大家介绍Lua中__index和__newindex之间的沉默与合作,包括了Lua中__index和__newindex之间的沉默与合作的使用技巧和注意事项,需要的朋友参考一下 因为不想在一篇文章里挤太多知识点,所以,有些小知识点就集合到这样的文章里吧~ 1.沉默技能——拒绝__index和__newindex效果 虽然__index和__newindex是很好用的功能,但是,有时候我们又希
本文向大家介绍详解Lua中的if语句的使用方法,包括了详解Lua中的if语句的使用方法的使用技巧和注意事项,需要的朋友参考一下 if语句由一个或多个语句组成一个布尔表达式。 语法 Lua编程语言的if语句语法是: 如果布尔表达式的计算结果为代码的if语句为true,那么块将被执行。如果if语句的末尾(右大括号后)布尔表达式计算为false,那么第一组代码将被执行。 Lua程序设计语言假定布尔tr