什么是Metatable

metatable是Lua中的重要概念,每一个table都可以加上metatable,以改变相应的table的行为。

Metatables举例

-- 声明一个正常的关系变量
lo_table = {} -- 声明空元表变量
lo_meta_table = {} -- 为关系变量t设置元表变量
setmetatable(lo_table, lo_meta_table) -- 获取一个关系变量的元表变量
getmetatable(lo_table)

上边的代码也可以写成一行,如下所示

-- setmetatable函数的返回值,就是该函数的第一个参数
lo_table = setmetatable({}, {})

创建复杂的元表变量

metatable可以包括任何东西,metatable特有的键一般以__开头,例如__index__newindex,它们的值一般是函数或其他table。

lo_table = setmetatable({}, {
__index = function(lo_table, key)
if key == "foo" then
return 0
else
return table[key]
end
end
})

__index

  这是metatable最常用的键了。

  当你通过键来访问table的时候,如果这个键没有值,那么Lua就会寻找该table的metatable(假定有metatable)中的__index键。如果__index包含一个表格,Lua会在表格中查找相应的键。

-- 创建元表变量
lo_meta_table = { name = "蓝鸥" } -- 设置该元表变量作为关系变量的
lo_table = setmetatable({}, { __index = lo_meta_table }) -- 打印lo_table变量的姓名 蓝鸥
print(lo_table.name) -- 打印lo_table变量年龄 nil
print(lo_table.age)

  如果__index包含一个函数的话,Lua就会调用那个函数,table和键会作为参数传递给函数。

-- 创建元表变量
lo_meta_table = {
name = "蓝鸥" ,
action = function ( param )
-- body
if(param == "学生") then
print("让教育回归本质")
else
print("让蓝鸥维护教育")
end
end
} -- 设置该元表变量作为关系变量的
lo_table = setmetatable({}, { __index = lo_meta_table }) -- 打印lo_table变量的动作 让教育回归本质
print(lo_table.action("学生")) -- 打印lo_table变量的动作 让蓝鸥维护教育
print(lo_table.action("肖浩")) -- 打印lo_table变量年龄 nil
print(lo_table.age)

__newindex

类似__index__newindex的值为函数或table,用于按键赋值的情况。

-- 创建元表变量
lo_meta_table = {} -- 设置该元表变量作为关系变量的
lo_table = setmetatable({}, { __newindex = lo_meta_table }) -- 设置lo_table变量的name关键字的值
lo_table.name = "蓝鸥" -- 打印lo_meta_table元表变量name关键字的值值
print(lo_meta_table.name) -- 打印lo_table变量name关键字的值
print(lo_table.name)
-- 创建元表变量
lo_meta_table = {} -- 设置该元表变量作为关系变量的
lo_table = setmetatable({}, { __newindex = function(t, key, value)
if type(value) == "number" then
rawset(t, key, value * value)
else
rawset(t, key, value)
end
end
}) -- 设置lo_table变量的name关键字的值
lo_table.name = "蓝鸥" -- 设置lo_table变量的age关键字的值
lo_table.age = 3 -- 打印lo_meta_table元表变量name关键字的值值
print(lo_meta_table.name) -- 打印lo_table变量name关键字的值
print(lo_table.name) -- 打印lo_meta_table元表变量age关键字的值值
print(lo_meta_table.age) -- 打印lo_table变量age关键字的值
print(lo_table.age)

上面的代码中使用了rawgetrawset以避免死循环。使用这两个函数,可以避免Lua使用__index__newindex

运算符

利用metatable可以定义运算符,例如+

-- 创建重载+号行为的表变量
lo_table = setmetatable({ 1, 2, 3 }, {
__add = function(lo_table, other)
new = {} -- 遍历元素加other
for _, v in ipairs(lo_table)
do table.insert(new, v + other)
end return new
end
}) -- 进行计算+
lo_table = lo_table + 2 -- 打印得到的结果
print(lo_table[1])
print(lo_table[2])
print(lo_table[3])

__index__newindex不同,__mul的值只能是函数。与__mul类似的键有:

  • __add (+)
  • __sub (-)
  • __div (/)
  • __mod (%)
  • __unm 取负
  • __concat (..)
  • __eq (==)
  • __lt (<)
  • __le (<=)

__call

__call使得你可以像调用函数一样调用table

t = setmetatable({}, {
__call = function(t, a, b, c, whatever)
return (a + b + c) * whatever
end
}) local result = t(1, 2, 3, 4)
print(result)

__tostring

最后讲下__tostring,它可以定义如何将一个table转换成字符串,经常和 print 配合使用,因为默认情况下,你打印table的时候会显示 table: 0x7f86f3d04d80 这样的代码

lo_table = setmetatable({ 1, 2, 3 }, {
__tostring = function(lo_table)
sum = 0
for _, v in pairs(lo_table)
do
sum = sum + v
end return "计算的结果是: " .. sum
end
}) -- prints out "计算的结果是: 6"
print(lo_table)

创建一个简单的向量Vector类

Vector = {}
Vector.__index = Vector function Vector.new(x, y)
return setmetatable({ x = x or 0, y = y or 0 }, Vector)
end -- __call关键字
setmetatable(Vector, { __call = function(_, ...) return Vector.new(...) end }) -- + 运算符
function Vector:__add(other)
-- ...
local result = Vector(self.x + other.x,self.y + other.y) return result
end -- __tostring关键字
function Vector:__tostring()
-- body return "x: " .. self.x .. " y: " .. self.y end a = Vector.new(12, 10)
b = Vector(20, 11)
c = a + b print(a)
print(c)

转自:http://www.cnblogs.com/daxiaxiaohao/p/4651767.html

Lua高级教程Metatables的更多相关文章

  1. [Lua]Lua高级教程Metatables

    什么是Metatable metatable是Lua中的重要概念,每一个table都可以加上metatable,以改变相应的table的行为. Metatables举例 -- 声明一个正常的关系变量 ...

  2. ios cocopods 安装使用及高级教程

    CocoaPods简介 每种语言发展到一个阶段,就会出现相应的依赖管理工具,例如Java语言的Maven,nodejs的npm.随着iOS开发者的增多,业界也出现了为iOS程序提供依赖管理的工具,它的 ...

  3. 【读书笔记】.Net并行编程高级教程(二)-- 任务并行

    前面一篇提到例子都是数据并行,但这并不是并行化的唯一形式,在.Net4之前,必须要创建多个线程或者线程池来利用多核技术.现在只需要使用新的Task实例就可以通过更简单的代码解决命令式任务并行问题. 1 ...

  4. 【读书笔记】.Net并行编程高级教程--Parallel

    一直觉得自己对并发了解不够深入,特别是看了<代码整洁之道>觉得自己有必要好好学学并发编程,因为性能也是衡量代码整洁的一大标准.而且在<失控>这本书中也多次提到并发,不管是计算机 ...

  5. 分享25个新鲜出炉的 Photoshop 高级教程

    网络上众多优秀的 Photoshop 实例教程是提高 Photoshop 技能的最佳学习途径.今天,我向大家分享25个新鲜出炉的 Photoshop 高级教程,提高你的设计技巧,制作时尚的图片效果.这 ...

  6. 展讯NAND Flash高级教程【转】

    转自:http://wenku.baidu.com/view/d236e6727fd5360cba1adb9e.html 展讯NAND Flash高级教程

  7. 转:Lua简明教程

    需要注意的是:lua中的变量如果没有特殊说明,全是全局变量,那怕是语句块或是函数里. 这里很奇怪,为什么在函数内部声明的变量默认也是global的呢? 函数的返回值 和Go语言一样,可以一条语句上赋多 ...

  8. Net并行编程高级教程--Parallel

    Net并行编程高级教程--Parallel 一直觉得自己对并发了解不够深入,特别是看了<代码整洁之道>觉得自己有必要好好学学并发编程,因为性能也是衡量代码整洁的一大标准.而且在<失控 ...

  9. [转帖]tar高级教程:增量备份、定时备份、网络备份

    tar高级教程:增量备份.定时备份.网络备份 作者: lesca 分类: Tutorials, Ubuntu 发布时间: 2012-03-01 11:42 ė浏览 27,065 次 61条评论 一.概 ...

随机推荐

  1. [SQL基础教程] 4-4 事务

    [SQL基础教程] 4 数据更新 4-4 事务 事务 需要在同一处理单元中执行的一系列更新处理的集合 创建事务 事务开始语句; DML语句1; DML语句2; . . . 事务结束语句; 事务开始语句 ...

  2. Python操作redis、memcache和ORM框架_Day13

    一.memcache Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站的速 ...

  3. memcahced缓存特点

    1.key-value数据结构 2.所有数据保存在内存中 3.可以分布式集群 4.处理并发的机制是libevent事件机制 5.当内容容量达到指定值后,就基于LRU(Least Recently Us ...

  4. JavaSE中主要的类与方法

    1.instanceof:测试左边的对象是否是右边类的实例;  数组.length:返回的是数组的长度; 2.Scanner类     nextInt():读取键盘录入的整数;  next():读取键 ...

  5. on事件绑定阻止冒泡的问题

    当使用on进行事件绑定时当要给document绑定click,而子元素要禁止冒泡,那么子元素里面的子元素的click事件就会无效了, 下面无效版: $('#queue').on('click', '. ...

  6. 安装oracle 10g时提示:操作系统版本: 必须是5.1或者5.2 怎么办?

    1.在安装目录中搜索refhost.xml,然后在适当位置添加以下内容,注意括号配对 <!--Microsoft Windows 7--> <OPERATING_SYSTEM> ...

  7. 使用ab对站点进行压力测试

    测试指令: window下: E:\wamp\bin\apache\Apache2.2.21\bin> .\ab -V  //查看是否按照了ab:(V 大写) E:\wamp\bin\apach ...

  8. 关于IIS和.NET 4.0的问题总结(转)

    注册asp.net 4.0 到iis   如果没有按照正常的先装iis后装.net的顺序,可以使用此命令重新注册一下 运行->cmd Microsoft Windows [版本 6.1.7601 ...

  9. Python学习笔记——基础篇【第六周】——Subprocess模块

    执行系统命令 可以执行shell命令的相关模块和函数有: os.system os.spawn* os.popen*          --废弃 popen2.*           --废弃 com ...

  10. redis写shell与ssh免密码登陆

     redis-cli参数:-h :指定要连接的主机IP或域名-p :指定连接的端口-a :指定密码-r :执行指定的命令-n :数据库名-x :将最后一个参数输出为value redis写shell- ...