Lua中的元表(metatable)、元方法(metamethod)详解

终于到了在实际中经常要用到的内容了——元表与元方法。

在第一次看见这两样东西的时候,可能会觉得它很深奥,但其实很好理解,虽然实际上它可能真的很深奥。(小若:停!滚粗。)

1.知道为什么1 + 1 = 2吗?

为什么在Lua中,1+1会等于2呢?(小若:难道除了Lua,其他地方就不等于2了?)
为什么数字和数字相加是合法的,为什么table和table相加就会报错?大家有想过这些问题吗?
 
没错,规则,这一切都只是规则而已,Lua规定了数字之间可以进行加减乘除,而table之间则不可以。

这是因为,在人类世界里,并没有table和table相加的概念。
 
而在Lua中,进行这些规则限定的秘密就在于元表和元方法。

2.元方法

元方法,听起来很深奥,其实它就是格子类型变量之间进行特殊操作的函数。
比如,数字的相加,它可能仅仅是一个函数。

比如:1+1  ,在底层里,它可能是这样的:add(1, 1)。而add函数就是用来计算两个数字间相加的结果。

再如:10X15,它可能是这样的:mul(10, 15)。mul函数就能返回两个数字相乘的结果。
(可能这例子不太恰当,但就是这么个意思~)
 
最后,如果是两个table呢?

代码如下:

local t1 = {};
local t2 = {};
t1 + t2;

它可能就是这样的:????
没错,Lua中不存在可以计算两个table相加的函数,也就是说,不存在这样的元方法。

3.元表

元表本身并没有什么作用,它是用来存放元方法的一个table。

Lua中的每一个值都有或者可以有一个元表,table和userdata可以各种拥有独立的元表。
但是,其他类型的值就只能共享其类型所属的元表,比如,数字,所有的数字都共用一个元表。

4.改变规则

如果说,我们就是希望将两个table进行相加呢?
试试看,如下代码:

代码如下:

local t1 = {};
    local t2 = {};
    local result = t1 + t2;

直接运行肯定报错的。

因此,为了满足我们这种需求,Lua允许我们修改元表。

一个元表,其实就是一个table值,所以,我们只需要新建一个table,添加元方法即可。

比如加法运算的元方法就是:__add,这是Lua规定的。

只要某个值的元表里含有__add这个元方法,那就可以使用+号进行运算。
 
如下代码:

代码如下:

-- 创建一个元表
    local mt = {};
    mt.__add = function(t1, t2)
        print("两个table相加的结果就是...神经病啊!table有什么好相加的啊!");
    end
    local t1 = {};
    local t2 = {};
  
    -- 给两个table设置新的元表
    setmetatable(t1, mt);
    setmetatable(t2, mt);
  
    -- 进行加法操作
    local result = t1 + t2;

首先创建了一个table变量mt,给这个table新增一个元素__add,这个table就拥有了作为元表的资格了。

然后创建两个新的table变量,使用setmetatable函数给table设置新的元表,此时,两个table变量就以mt作为元表了。

最后,对t1和t2进行加法操作,这时就会从元表中查找__add元方法,如果找到的话,就调用这个元方法对两个变量进行加法操作。
 
输出结果如下:

代码如下:

[LUA-print] 两个table相加的结果就是…神经病啊!table有什么好相加的啊!

就是这么简单,元表和元方法其实就是给Lua里的值设定一些操作,比如加法、减法之类的,让我们可以对这些操作自定义。
 
不过,有几点要特别注意的:
a.创建一个新的table变量时,它是不存在元表的(可以用getmetatable函数获取某个对象的元表,就能知道这个对象有没有元表存在了)

b.在Lua中,只能设置table的元表,其他类型的值的元表,只能通过C代码来完成

5.结束

好吧,其实我觉得这篇文章写得比较乱,不是很满意。

我想以一种更简单的方式来介绍元表和元方法,不过,似乎适得其反了。

没关系了,希望大家勉强能看明白~

本篇只是作为元表和元方法的概述,后面还会详细介绍更多的内容。

(0)

相关推荐

  • 举例说明Lua中元表和元方法的使用

    table的元表提供了一种机制,可以重定义table的一些操作. 之后我们会看到元表是如何支持类似js的prototype行为. 复制代码 代码如下: f1 = {a = 1, b = 2}  -- 表示一个分数 a/b. f2 = {a = 2, b = 3} 复制代码 代码如下: -- 这个是错误的: -- s = f1 + f2 metafraction = {} function metafraction.__add(f1, f2)   sum = {}   sum.b = f1.b *

  • Lua中的元表与元方法学习总结

    前言 元表对应的英文是metatable,元方法是metamethod.我们都知道,在C++中,两个类是无法直接相加的,但是,如果你重载了"+"符号,就可以进行类的加法运算.在Lua中也有这个道理,两个table类型的变量,你是无法直接进行"+"操作的,如果你定义了一个指定的函数,就可以进行了.那这篇博文就是主要讲的如何定义这个指定的函数,这个指定的函数是什么?希望对学习Lua的朋友有帮助. Lua是怎么做的? 通常,Lua中的每个值都有一套预定义的操作集合,比如数

  • Lua教程(九):元表与元方法详解

    Lua中提供的元表是用于帮助Lua数据变量完成某些非预定义功能的个性化行为,如两个table的相加.假设a和b都是table,通过元表可以定义如何计算表达式a+b.当Lua试图将两个table相加时,它会先检查两者之一是否有元表,然后检查该元表中是否存在__add字段,如果有,就调用该字段对应的值.这个值就是所谓的"元方法",这个函数用于计算table的和. Lua中每个值都有一个元表.table和userdata可以有各自独立的元表,而其它数据类型的值则共享其类型所属的单一元表.缺省

  • Lua中的元表(metatable)、元方法(metamethod)详解

    终于到了在实际中经常要用到的内容了--元表与元方法. 在第一次看见这两样东西的时候,可能会觉得它很深奥,但其实很好理解,虽然实际上它可能真的很深奥.(小若:停!滚粗.) 1.知道为什么1 + 1 = 2吗? 为什么在Lua中,1+1会等于2呢?(小若:难道除了Lua,其他地方就不等于2了?) 为什么数字和数字相加是合法的,为什么table和table相加就会报错?大家有想过这些问题吗?   没错,规则,这一切都只是规则而已,Lua规定了数字之间可以进行加减乘除,而table之间则不可以. 这是因

  • Lua中的元表和元方法学习笔记

    元表(metatable)是 Lua 里每种类型的值的默认操作方式的集合,例如,数字可以加减乘除.字符串可以连接合并.table 可以插入一对 key-value 值.函数可以被调用等等,这些操作都遵循其预定义的行为来执行. 而值的默认操作方式不是一成不变的,可以通过元表来修改其行为表现,或者是新定义一些默认没有的操作.例如,当两个 table 相加时, Lua 会检查它们之间的元表里是否有 "__add" 这个函数,如果定义有这个函数, 则调用这个函数来执行一次加法操作. 这里,相加

  • Oracle表中重复数据去重的方法实例详解

    Oracle表中重复数据去重的方法实例详解 我们在项目中肯定会遇到一种情况,就是表中没有主键 有重复数据 或者有主键 但是部分字段有重复数据 而我们需要过滤掉重复数据 下面是一种解决方法 delete from mytest ms where rowid in (select aa.rid from (select rowid as rid, row_number() over(partition by s.name order by s.id) as nu from mytest s) aa

  • PHP中Closure类的使用方法及详解

    Closure,匿名函数,又称为Anonymous functions,是php5.3的时候引入的.匿名函数就是没有定义名字的函数.这点牢牢记住就能理解匿名函数的定义了. Closure 类(PHP 5 >= 5.3.0)简介 用于代表 匿名函数 的类. 匿名函数(在 PHP 5.3 中被引入)会产生这个类型的对象,下面我们来看一下PHP Closure类的使用方法及介绍. PHP Closure类之前在PHP预定义接口中介绍过,但它可不是interface哦,它是一个内部的final类.Clo

  • 在Spring Boot应用程序中使用Apache Kafka的方法步骤详解

    第1步:生成我们的项目: Spring Initializr来生成我们的项目.我们的项目将提供Spring MVC / Web支持和Apache Kafka支持. 第2步:发布/读取Kafka主题中的消息: <b>public</b> <b>class</b> User { <b>private</b> String name; <b>private</b> <b>int</b> age

  • Vue中遍历数组的新方法实例详解

    1.foreach foreach循环对不能使用return来停止循环 search(keyword){ var newList = [] this.urls.forEach(item =>{ if(item.name.indexOf(keyword) != -1){ newList.push(item) } }) return newList } 2.filter item对象就是遍历数组中的一个元素,includes是es6中的新方法,在search方法中直接返回新数组 search(key

  • Vue中使用webpack别名的方法实例详解

    在工作中,我们经常会写出这种代码: import MHeader from '../../components/m-header/m-header' @import "../../common/stylus/variable" @import "../../common/stylus/mixin" 即,需要引入公共文件,但是公共文件的文件路径里当前文件很远,那么就会形成上面示例中的那种路径很长的情况. 而因为文件目录是约定俗成的,不可轻易更改,无法修改相对路径.那么

  • Android Studio 中运行 groovy 程序的方法图文详解

    Groovy简介 Groovy是一种基于JVM(Java虚拟机)的敏捷开发语言,它结合了Python.Ruby和Smalltalk的许多强大的特性,Groovy 代码能够与 Java 代码很好地结合,也能用于扩展现有代码.由于其运行在 JVM 上的特性,Groovy也可以使用其他非Java语言编写的库. Groovy 是 用于Java虚拟机的一种敏捷的动态语言,它是一种成熟的面向对象编程语言,既可以用于面向对象编程,又可以用作纯粹的脚本语言.使用该种语言不必编写过多的代码,同时又具有闭包和动态语

  • Lua中的元方法__newindex详解

    好吧,我写文章的进度已经赶不上看书的进度了,简单的几段文字就够我唠叨一篇文章了. 今天继续来说说元方法,与__index有点相似的__newindex元方法. 1.查询与更新 上一篇文章我们介绍了__index元方法,总结来说,__index元方法是用于处理调用table中不存在的字段. 注意,[调用]这个词,只是调用,而不是赋值.   如果,我们要对table中某个不存在的字段赋值呢?(小若:就,直接赋值啊!) 没错,我们直接就能赋值了,不会报错的. 问题是,如果我想监控这个操作呢?如果有人想

随机推荐