举例说明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 * f2.b
  sum.a = f1.a * f2.b + f2.a * f1.b
  return sum
end

setmetatable(f1, metafraction)
setmetatable(f2, metafraction)

s = f1 + f2  -- 调用在f1的元表上的__add(f1, f2) 方法

-- f1, f2 没有能访问它们元表的key,这与prototype不一样,
-- 所以你必须用getmetatable(f1)去获得元表。元表是一个普通的table,
-- Lua可以通过通常的方式去访问它的key,例如__add。

代码如下:

-- 不过下面的代码是错误的,因为s没有元表:
-- t = s + s
-- 下面的类形式的模式可以解决这个问题:

-- 元表的__index 可以重载点运算符的查找:
defaultFavs = {animal = 'gru', food = 'donuts'}
myFavs = {food = 'pizza'}
setmetatable(myFavs, {__index = defaultFavs})
eatenBy = myFavs.animal  -- 可以工作!这要感谢元表的支持

如果在table中直接查找key失败,会使用元表的__index 继续查找,并且是递归的查找

__index的值也可以是函数function(tbl, key) ,这样可以支持更多的自定义的查找。

__index、__add等等,被称为元方法。
这里是table的元方法的全部清单:

-- __add(a, b)                     for a + b
-- __sub(a, b)                     for a - b
-- __mul(a, b)                     for a * b
-- __div(a, b)                     for a / b
-- __mod(a, b)                     for a % b
-- __pow(a, b)                     for a ^ b
-- __unm(a)                        for -a
-- __concat(a, b)                  for a .. b
-- __len(a)                        for #a
-- __eq(a, b)                      for a == b
-- __lt(a, b)                      for a < b
-- __le(a, b)                      for a <= b
-- __index(a, b)  <fn or a table>  for a.b
-- __newindex(a, b, c)             for a.b = c
-- __call(a, ...)                  for a(...)

类风格的table和继承

类并不是内置的;有不同的方法通过表和元表来实现。

下面是一个例子,后面是对例子的解释

代码如下:

Dog = {}                                   -- 1.

function Dog:new()                         -- 2.
  newObj = {sound = 'woof'}                -- 3.
  self.__index = self                      -- 4.
  return setmetatable(newObj, self)        -- 5.
end

function Dog:makeSound()                   -- 6.
  print('I say ' .. self.sound)
end

mrDog = Dog:new()                          -- 7.
mrDog:makeSound()  -- 'I say woof'         -- 8.

-- 1. Dog看上去像一个类;其实它完全是一个table。
-- 2. 函数tablename:fn(...) 与函数tablename.fn(self, ...) 是一样的
--    冒号(:)只是添加了self作为第一个参数。
--    下面的第7和第8条说明了self变量是如何得到其值的。
-- 3. newObj是类Dog的一个实例。
-- 4. self为初始化的类实例。通常self = Dog,不过继承关系可以改变这个。
--    如果把newObj的元表和__index都设置为self,
--    newObj就可以得到self的函数。
-- 5. 记住:setmetatable返回其第一个参数。
-- 6. 冒号(:)在第2条是工作的,不过这里我们期望
--    self是一个实例,而不是类
-- 7. 与Dog.new(Dog)类似,所以 self = Dog in new()。
-- 8. 与mrDog.makeSound(mrDog)一样; self = mrDog。

 继承的例子:

代码如下:

LoudDog = Dog:new()                           -- 1.

function LoudDog:makeSound()
  s = self.sound .. ' '                       -- 2.
  print(s .. s .. s)
end

seymour = LoudDog:new()                       -- 3.
seymour:makeSound()  -- 'woof woof woof'      -- 4.

-- 1. LoudDog获得Dog的方法和变量列表。
-- 2. 通过new(),self有一个'sound'的key from new(),参见第3条。
-- 3. 与LoudDog.new(LoudDog)一样,并且被转换成
--    Dog.new(LoudDog),因为LoudDog没有'new' 的key,
--    不过在它的元表可以看到 __index = Dog。
--    结果: seymour的元表是LoudDog,并且
--    LoudDog.__index = LoudDog。所以有seymour.key
--    = seymour.key, LoudDog.key, Dog.key, 要看
--    针对给定的key哪一个table排在前面。
-- 4. 在LoudDog可以找到'makeSound'的key;这与
--    LoudDog.makeSound(seymour)一样。

代码如下:

-- 如果需要,子类也可以有new(),与基类的类似:
function LoudDog:new()
  newObj = {}
  -- 初始化newObj
  self.__index = self
  return setmetatable(newObj, self)
end

(0)

相关推荐

  • 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中元表和元方法的使用

    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中强大的元方法__index详解

    今天要来介绍比较好玩的内容--__index元方法 1.我是备胎,记得回头看看 咳咳,相信每一位女生都拥有或者不知不觉中拥有了一些备胎,啊,当然,又或许是成为过别人的备胎. 没有备胎的人,就不是完整的人生.(小若:停!)   我们来想象一下,如果对一个table进行取值操作,但是table根本就没有这个值呢? 比如: 复制代码 代码如下: local t = {         name = "hehe",     }     print(t.money); 输出结果当然是:nil t

  • Lua元表与元方法实例讲解

    Lua中提供的元表(metatable)与元方法(metamethod)是一种非常重要的语法,metatable主要用于做一些类似于C++重载操作符式的功能. Lua中提供的元表是用于帮助lua变量完成某些非预定义功能的个性化行为,如两个table的相加,通过让两者指向同一元表并修改该元表的元方法可以实现该功能. 任何table都可以成为任何值的元表,而一组相关的table也可以共享一个元表. 一些MetaMethod: 复制代码 代码如下: __add(a, b)               

  • Lua中类的实现原理探讨(Lua中实现类的方法)

    Lua中没有类的概念,但我们可以利用Lua本身的语言特性来实现类. 下文将详细的解释在Lua中实现类的原理,涉及到的细节点将拆分出来讲,相信对Lua中实现类的理解有困难的同学将会释疑. 类是什么? 想要实现类,就要知道类到底是什么. 在我看来,类,就是一个自己定义的变量类型.它约定了一些它的属性和方法,是属性和方法的一个集合. 所有的方法都需要一个名字,即使是匿名函数实际上也有个名字.这就形成了方法名和方法函数的键值映射关系,即方法名为键,映射的值为方法函数. 比如说有一个类是人,人有一个说话的

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

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

  • Lua中模块以及实现方法指南

    从使用的角度来看,一个模块就是一个程序库,可以通过Lua自身提供的require来加载.然后便得到一个全局变量,表示一个table.这个table就是像一个名字空间,其内容就是模块导出的所有东西,例如函数和常量.简单的说,Lua中的模块就是一个table,table中可以包括任何东西.本文首先详细介绍模块相关的require函数,包括该函数的执行流程以及查找模块的路径,然后介绍了实现模块的三种方法,并给出相应的优缺点.  require函数 该函数用来加载一个模块,即按指定的路径和传入的参数,查

  • 举例讲解Ruby中require的使用方法

    同一目录下的文件,如/usr/local/ruby/foo.rb与/usr/local/ruby/bar.rb两个文件. 如果直接在foo.rb中 require 'bar' 执行时会报找不到bar.rb错误. 这是因为运行 /home/oldsong$ ruby /usr/local/ruby/foo.rb 时会在ruby安装的lib目录和/home/oldsong/目录下查找bar.rb.而不会去rb文件的目录/usr/local/ruby/下查找. 所以除引用系统rb外,require中不

  • 举例讲解Lua中的Table数据结构

    文中-- 两个横线开始单行的注释,--[[加上两个[和]表示多行的注释--]]. 复制代码 代码如下: -- Table = Lua唯一的数据结构; --         它们是关联数组. -- 类似于PHP的数组或者js的对象, -- 它们是哈希查找表(dict),也可以按list去使用. 复制代码 代码如下: -- 按字典/map的方式使用Table: -- Dict的迭代默认使用string类型的key: t = {key1 = 'value1', key2 = false} 复制代码 代

随机推荐