Metatable:

  lua中的每一个表都有其Metatable,默认情况下Metatable为nil。可通过setmetatable函数设置或者改变一个表的Metatable,

也可以通过getmetatable得到一个表的Metatable。任何一个表都可以是其它表的Metatable,可以多个表共享一个Metatable,

一个表也可以是自身的Metatable。

作用:使我们可以改变table的行为

  在metatable中设置__add, __sub, __mul, __div, __eq(等于), __lt(小于), __le(小于等于), __unm(负), __pow(幂),

__concat (定义连接行为),__tostring等metamethod方法,可以实现类似于C++中运算符重载的效果。

实例:

 Set = {}
Set.mt = {}
--setmetatable(Set.mt, Set.mt) function Set:new( t )
local tb = t or {}
setmetatable(tb, Set.mt)
return tb
end function Set.union( a, b )
local res = Set.new()
for _, k in pairs(a) do
table.insert(res, k)
end
for _, k in pairs(b) do
table.insert(res, k)
end
return res
end function Set.tostring( set )
local s = "{\n"
local sep = " "
for _, k in pairs(set) do
if type(k) == "boolean" then
k = "boolean"
elseif type(k) == "function" then
k = "function"
elseif type(k) == "table" then
-- k = "table"
k = Set.tostring(k)
end
s = s.." [".._.."]".." = "..k.."\n"
end
return s.."}"
end local s1 = Set:new({, , , })
local s2 = Set:new({, }) Set.mt.__add = Set.union
Set.mt.__concat = Set.union
Set.mt.__tostring = Set.tostring local s3 = s1 + s2
print(s3) local s4 = s1..s2
print(s4) 这里+和..是等价的,所以s3==s4,结果:
{
[] =
[] =
[] =
[] =
[] =
[] =
}
{
[] =
[] =
[] =
[] =
[] =
[] =
}

  当我们对两个有不同metatable的表进行加操作时,则检查第一个表的metatabled是否有__add,有则用之,
没有就再去检查第二个表。。。还没有就报错。

__index:

  我们在访问一个表不存在的域时,lua解释器会去查找metatabled中是否有__index方法(metamethod),如果不存在则返回nil,

否则由__index方法返回。__index可以是函数也可以是表

 Set.mt.__index = function (tb, key)--这里会传进来一个table和key
return "get "..key.." is null"
end -- Set.mt.__index = {sex = "man"} local myTb = Set:new({name = "Mical", age = })
print(myTb.name)
print(myTb.age)
print(myTb.sex) 结果:
Mical get sex is null

__newindex:

  当你给表不存在的值赋值时,lua解释器就会查找对应Metatable中的__newindex方法(metamethod)。如果存在则调用这个函数而不进行赋值。

 Set.mt.__newindex = function (tb, key, val)
print("can't set "..key.." value")
end
local myTb = Set:new({name = "Mical", age = })
myTb.sex = "man" 结果:
can't set sex value

rawset(t, k, v):

  不调用任何的metamethod 对表t的k域赋值为v,上述例子就可以用rawset(myTb, "sex", "man")赋值。

给表设置默认值:

 local key = {}
local mt = {__index = function(t) return t[key] end}
function setDefault(t, d)
t[key] = d
setmetatable(t, mt)
end

监控表: 

index用于查询,__newindex用于更新,它们都是在访问的域不存在的时候才起作用。如果我们想捕获对一个表的所有反问,可以通过保持这个表为空来实现。

 local index = {}
local mt = { __index = function(t, k)
print("access to element "..tostring(k))
return t[index][k]
end,
__newindex = function(t, k, v)
print("update of element "..tostring(k).." to "..tostring(v))
t[index][k] = v
end
}
local function track(t)
local proxy = {}
proxy[index] = t
setmetatable(proxy, mt)
return proxy
end local myTb = {}
local nowTb = track(myTb)
nowTb.name = "mical"
print(nowTb.name) nowTb.age =
print(nowTb.sex) for k, v in pairs(nowTb) do
print(k, "\n", v);
end

lua的Metatables和Metamethods的更多相关文章

  1. Lua 与 Redis

    Lua 与 Redis 标签: Java与NoSQL 从 2.6版本 起, Redis 开始支持 Lua 脚本 让开发者自己扩展 Redis - 案例-实现访问频率限制: 实现访问者 $ip 在一定的 ...

  2. [转]LUA元表

    lua元表和元方法 <lua程序设计> 13章 读书笔记 lua中每个值都有一个元表,talble和userdata可以有各自独立的元表,而其它类型的值则共享其类型所属的单一元表.lua在 ...

  3. Lua.LearningLua.7-userdata

    Learning Lua: 7 - userdata 1. Userdata basic "There are eight basic types in Lua: nil, boolean, ...

  4. 初识lua

    转自:http://www.oschina.net/question/12_115993-- 两个横线是单行注释(译者注:这跟 SQL 一样) --[[ 增加两个 [ 和 ] 变成多行注释 我是多行注 ...

  5. lua学习

    在lua中,一切都是变量,除了关键字. 1.注释: 单行注释: 连续两个减号“--”表示注释的开始,一直延续到行末.相当于C语言中的“//” 多行注释:由“--[[”表示注释开始, “]]”表示注释结 ...

  6. Lua语言模型 与 Redis应用

    Lua语言模型 与 Redis应用 标签: Java与NoSQL 从 2.6版本 起, Redis 开始支持 Lua 脚本 让开发者自己扩展 Redis. 本篇博客主要介绍了 Lua 语言不一样的设计 ...

  7. Learn Lua in 15 Minutes

    原文地址:http://tylerneylon.com/a/learn-lua/ Learn Lua in 15 Minutes more or less For a more in-depth Lu ...

  8. Redis结合Lua脚本实现高并发原子性操作

    从 2.6版本 起, Redis 开始支持 Lua 脚本 让开发者自己扩展 Redis … 案例-实现访问频率限制: 实现访问者 $ip 在一定的时间 $time 内只能访问 $limit 次. 非脚 ...

  9. 15分钟入门lua

    目录:[ - ] -- 1. Variables and flow control. -- 2. Functions. -- 3. Tables. -- 3.1 Metatables and meta ...

随机推荐

  1. idea搭建第一个springboot

    1.打开idea开发工具,在菜单栏选择File-->New-->Project...-->Spring Initializer说明:社区版的idea是没有Spring Initial ...

  2. SP1825 FTOUR2 - Free tour II 点分治+启发式合并+未调完

    题意翻译 给定一棵n个点的树,树上有m个黑点,求出一条路径,使得这条路径经过的黑点数小于等于k,且路径长度最大 Code: #include <bits/stdc++.h> using n ...

  3. PAT_A1134#Vertex Cover

    Source: PAT A1134 Vertex Cover (25 分) Description: A vertex cover of a graph is a set of vertices su ...

  4. 【剑指Offer】58、对称的二叉树

      题目描述:   请实现一个函数,用来判断一颗二叉树是不是对称的.注意,如果一个二叉树同此二叉树的镜像是同样的,定义其为对称的.   解题思路:   本题判断一棵树是不是对称的,和第18题可以对比分 ...

  5. mysql字符集和排序规则

    1.关于字符集和排序规则所为字符集,就是用来定义字符在数据库中的编码的集合.常见的字符集有:utf8(支持中文)和AccIS(不支持中文) 数据库中的排序规则用来定义字符在进行排序和比较的时候的一种规 ...

  6. 【习题 4-7 UVA - 509】RAID!

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 如果一行里面某位有>1个x 那么是invalid的. 没有x的话. 可以分析以下(设输入的标准Even为0,然后Odd为1) ...

  7. setTimeout(fn,0)的作用分析

    众所周知,大家对setTimeout的用法肯定都比较熟悉了,但是不是还是会经常忘记使用呢,例如博主阿里面试时就忘了,见阿里前端面试. 今天跟大家讨论一下setTimeout(fn,0)的用法,相信很多 ...

  8. HDU 2415 Bribing FIPA

    Bribing FIPA Time Limit: 1000ms Memory Limit: 32768KB This problem will be judged on HDU. Original I ...

  9. TCP连接之未连接队列的理解

    tcp服务器在TCP/IP协议中,TCP协议提供可靠的连接服务,采用三次握手建立一个连接. 第一次握手:建立连接时,客户端发送syn包(syn=j)到服务器,并进入SYN_SEND状态,等待服务器确认 ...

  10. EXPLAIN sql优化方法(1) 添加索引

    添加索引优化器更高效率地执行语句 假设我们有两个数据表t1和t2,每个有1000行,包含的值从1到1000.下面的查询查找出两个表中值相同的数据行: mysql> SELECT t1.i1, t ...