Lua中__index和__newindex之间的沉默与合作_Lua

因为不想在一篇文章里挤太多知识点,所以,有些小知识点就集合到这样的文章里吧~

1.沉默技能——拒绝__index和__newindex效果

虽然__index和__newindex是很好用的功能,但是,有时候我们又希望很纯粹地去调用table或者给table赋值。

那,这时候怎么办?给table重新设置一个元表?不,这个做法很糟糕~

于是,体贴的Lua又给我们提供了这样的调用方式,如下代码:

复制代码 代码如下:

    local smartMan = {
        name = "none",
    }
  
    local t1 = {
        hehe = 123;
    };
  
    local mt = {
        __index = smartMan,
        __newindex = function(t, k, v)
            print("别赋值!");
        end
    }
  
    setmetatable(t1, mt);
  
    print(rawget(t1, "name"));
    print(rawget(t1, "hehe"));
    rawset(t1, "name", "小偷");
    print(t1.name);

通过rawget函数可以忽略元表的__index功效,纯粹地从t1中调用字段。

rawget的第一个参数是要调用的table,第二个参数是table的字段名。

因此,通过rawget调用t1的name字段,只能返回nil,而调用hehe字段,则能正确取得值。

同样的是,rawset函数可以忽略元表的__newindex功效,纯粹地给t1赋值。

来看看输出结果:

复制代码 代码如下:

[LUA-print] nil
[LUA-print] 123
[LUA-print] 小偷

获取name字段,输出nil;
获取hehe字段,输出123;
修改name字段后,输出”小偷”

这就相当于t1并不存在__index和__newindex元方法了。
怎么样,这个沉默技能很有意思吧。

2.只读的table

呐,假设你又继续是一个主程,你写了一个很牛的功能,然后作为主程的你,每晚都要回家看电影。

所以你的功能不得不交给公司里那些刚毕业不到30年的新人去维护,让他们天天加班到晚上6点半。(小若:喂!6点半算加班吗?)

然而,这么牛的功能,可不能被这些新人随便改坏了,所以,除了保护table的元表之外,你还希望保护table的字段。

你要确保这些新人不会去修改你table的字段值。

没错,这时候就可以使用__index和__newindex来实现了,如下代码:

复制代码 代码如下:

local function readOnly(t)
    local newT = {};
    local mt = {
        __index = t,
        __newindex = function()
            error("别修改我!我是只读的!");
        end
    }
    setmetatable(newT, mt);
    return newT;
end
local days = readOnly({"星期一", "星期二", "星期日"});
  
days[2] = "星期三哪去了啊?" ;

这可能有点难弄懂,先来看看输出结果吧:

复制代码 代码如下:

[LUA-print] LUA ERROR: [string "src/main.lua"]:130: [string "src/main.lua"]:76: 别修改我!我是只读的!

没错,通过readOnly产生的table,是无法进行赋值操作的。
 
那么,原理呢?我们来一步步思考吧:

a.首先,readOnly会创建一个新的table,然后把我们传进去的table作为__index元方法。

b.元表里还增加了__newindex,用来阻止不存在字段的赋值操作。

c.readOnly返回的table已经不是我们原来的table了,它是一个空的table,但是它被设置了一个新的元表。

d.开始对days执行赋值操作:days[2] = “星期三哪去了啊?” 。

e.days是一个空的table,所以它不存在这个字段,也因此,会调用__newindex元方法,赋值失败。

f.如果只是调用days,不进行赋值,如:print(days[2]); 则能正常输出字段值,因为days的元表里有__index元方法。虽然days中不存在2这个字段,但是可以通过__index找到这个字段。
 
总而言之,最终,days成为了一个只可以读取,不能进行赋值操作的table。
(小若:那如果我使用rawset函数呢?不就打破了你的限制吗?)
咳咳,我们继续。

3.结束

终于结束,这几天几乎都在写文章了,没怎么看书,不过我会继续坚持写文章的~
看完书不记录一下,总觉得不够深刻~而且写完文章心情很好~

时间: 2024-07-31 04:50:41

Lua中__index和__newindex之间的沉默与合作_Lua的相关文章

Lua中的模块(module)和包(package)详解_Lua

前言 从Lua5.1版本开始,就对模块和包添加了新的支持,可是使用require和module来定义和使用模块和包.require用于使用模块,module用于创建模块.简单的说,一个模块就是一个程序库,可以通过require来加载.然后便得到了一个全局变量,表示一个table.这个table就像是一个命名空间,其内容就是模块中导出的所有东西,比如函数和常量,一个符合规范的模块还应使require返回这个table.现在就来具体的总结一下require和module这两个函数. require函

Lua中函数与面向对象编程的基础知识整理_Lua

函数 1. 基础知识调用函数都需要写圆括号,即使没有参数,但有一种特殊例外:函数若只有一个参数且参数是字面字符串或table构造式,则圆括号可有可无,如dofile 'a.lua',f{x=10, y=20}. Lua为面向对象式的调用提供冒号操作符的特殊语法,如o.foo(o, x)等价于o:foo(x).和Javascript类似,调用函数时提供的实参数量可以与形参数量不同,若实参多了则舍弃,不足则多余的形参初始化为nil. 1.1 多重返回值 Lua允许函数返回多个结果,函数返回如retu

解析Lua中的全局环境、包、模块组织结构_Lua

模块就是一个程序库,而包是一系列模块.Lua中可以通过require来加载模块,然后得到一个全局变量表示一个table.Lua将其所有的全局变量保存在一个被称为"环境"的常规table中.本文首先介绍环境的一些实用技术,然后介绍如何引用模块及编写模块的基本方法. 1. 环境Lua将环境table保存在一个全局变量_G中,可以对其访问和设置.有时我们想操作一个全局变量,而它的名称却存储在另一个变量中,或者需要通过运行时的计算才能得到,可以通过value = _G[varname]来获得动

Lua中实现sleep函数功能的4种方法_Lua

一个不幸的消息是Lua中没有内置sleep函数,我们需要DIY.有4种方法可以实现sleep函数,如下: 方法1 复制代码 代码如下: --在一个死循环中设置一个跳出条件,但是这样的做法会占用大量CPU资源,强烈不推荐使用哦 function sleep(n)    local t0 = os.clock()    while os.clock() - t0 <= n do end end 方法2 复制代码 代码如下: --调用系统的sleep函数,不消耗CPU,但是Windows系统中没有内置

Lua中使用table实现的其它5种数据结构_Lua

lua中的table不是一种简单的数据结构,它可以作为其他数据结构的基础,如:数组,记录,链表,队列等都可以用它来表示. 1.数组 在lua中,table的索引可以有很多种表示方式.如果用整数来表示table的索引,即可用table来实现数组,在lua中索引通常都会从1开始. 复制代码 代码如下: --二维数组 n=10 m=10 arr={} for i=1,n do      arr[i]={}    for j=1,m do       arr[i][j]=i*j    end end f

Lua中的for循环和迭代器的秘密探究_Lua

上一篇我们介绍了,可以使用for循环来完成迭代器的调用,十分简洁. 那么,具体这for循环做了什么呢?我当然没有去看源码,我只是看书而已. 资料来源于<Lua程序设计>第二版,如果这本书的内容没有错的话,那么,本篇文章理论上也不会有错~ 1.返回两个值的迭代器 pairs是能遍历table的key和value的,而我们之前写的dieDaiQi函数只能返回value. 所以,我们要改改dieDaiQi函数,如下: 复制代码 代码如下: function dieDaiQi(t)     local

Lua中函数的几个特别之处探究_Lua

没想到距离上一篇基础补充已经过了1年多了,最近准备捡回Lua,把基础都补补,今天来聊聊Lua的函数吧~ 0.环境 我突然对Lua又大感兴趣的最主要原因是,Cocos Code IDE开始浮出水面了,它是Cocos2d-x官方出的一款专门针对Cocos2d-x+Lua或JS的IDE,试着用了,虽然不能说很完美,但,很值得期待. 所以,本文使用的Lua编辑器就选它了,大家就随意吧~ 1.扫盲--Lua的函数 Lua要创建和调用函数都十分简单,如代码: 复制代码 代码如下: function muto

Lua中的loadfile、dofile、require详解_Lua

本来今天不应该讨论这几个函数的,不过,为了凑字数..不,为了方便以后的文章,还是要谈谈这几个基础函数的~ 1.loadfile--只编译,不运行 loadfile故名思议,它只会加载文件,编译代码,不会运行文件里的代码. 比如,我们有一个hellofile.lua文件: 复制代码 代码如下: print("hello"); function hehe() print("hello"); end 这个文件里有一句代码,和一个函数.试试用loadfile加载这个文件,如

Lua中遍历数组和table的4种方法_Lua

方法一,可以用for来遍历: 复制代码 代码如下: do      table_week = {      "w",      "e",      "r",      "t",      "y",      "u",      "i",      }        for i = 1, #table_week do          print(table_week[