13 Metatables and Metamethods
- metatable和metamethod可以允许我们对一个值做未定义的操作,Lua中的每个值都可以有一个metatable,table和userdata有自己的metatable,其他类型的值对于本类型的所有值共享一个metatable,Lua创建的表默认无metatable。
1
2t = {}
print(getmetatable(t)) --> nil
用setmetatable()
改变table的metatable:1
2
3t1 = {}
setmetatable(t, t1)
print(getmetatable(t) == t1) --> true
一个table可以是任何值的metatable,一组相关的table可以共享一个公共的metatable来描述它们公共的行为,一个table也可以做自己的metatable来描述自己的行为。
用算术metamethod对集合求并、求交
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41Set = {}
local mt = {}
-- create a new set with the values of a given list
function Set.new(l)
local set = {}
setmetatable(set, mt)
for _, v in ipairs(l) do set[v] = true end
return set
end
function Set.union(a, b)
local res = Set.new{}
for k in pairs(a) do res[k] = true end
for k in pairs(b) do res[k] = true end
return res
end
function Set.intersection(a, b)
local res = Set.new{}
for k in pairs(a) do
res[k] = b[k]
end
return res
end
-- presents a set as a string
function Set.tostring(set)
local l = {} -- list to put all elements from the set
for e in pairs(set) do
l[#l + 1] = e
end
return "{" .. table.concat(l, ", ") .. "}"
end
-- print a set
function Set.print(s)
print(Set.tostring(s))
end
s1 = Set.new{10, 20, 30, 50}
s2 = Set.new{30, 1}
mt.__add = Set.union
mt.__mul = Set.intersection
s3 = s1 + s2
Set.print(s3)
Set.print(s3 * s1)算术运算符在metatable中对应的域名:
field name | operator symbol | operator name |
---|---|---|
__add | + | addition |
__mul | * | multiplication |
__sub | - | subtraction |
__div | / | division |
__unm | - | negation |
__mod | % | modulo |
__pow | ^ | exponentiation |
__len | # | element number |
__concat | .. | concatenation |
- 对于一个算术运算,Lua从左到右查找操作数metatable中相应的metamethod,优先使用最先找到的,如果没找到则报错
1
2s = Set.new{1, 2, 3}
s = s + 8 --> bad argument #1 to 'pairs' (table expected, got number)
安全检查:1
2
3
4
5function Set.union(a, b)
if getmetatable(a) ~= mt or getmetatable(b) ~= mt then
error("attemp to 'add' a set with a non-set value", 2)
end
<as before>
- 关系运算符在metatable中对应的域名:
field name | operator symbol | operator name |
---|---|---|
__eq | = | equal to |
__lt | < | less than |
__le | <= | less than or equal to |
a~=b
变为not (a==b)
,a>b
变为b<a
,a>=b
变为b<=a
- 用关系metamethod对集合进行比较
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19mt.__le = function (a, b) -- set containment
for k in pairs(a) do
if not b[k] then return false end
end
return true
end
mt.__lt = function (a, b)
return a <= b and not (b <= a)
end
mt.__eq = function (a, b)
return a <= b and b <= a
end
s1 = Set.new{2, 4}
s2 = Set.new{4, 10, 2}
print(s1 <= s2)
print(s1 < s2)
print(s1 >= s1)
print(s1 > s2)
print(s1 == s2 * s1)
先构造__le
,再用其来构造其他关系运算
对于不同基础类型的两个对象或者有不同的metamethod,相等比较总是返回false而不会调用任何metamethod,因此一个集合总是不同与一个数,无论metamethod是什么
库定义metametod
print()
总是调用tostring()
来格式化输出,但是tostring()
会先检查值是否有__tostring
metamethod,如果有则将其结果返回1
2
3
4print({}) --> table: 0xda4680
mt.__tostring = Set.tostring
s1 = Set.new{10, 4, 5}
print(s1) --> {4, 5, 10}
j
setmetatable()
和getmetatable()
用一个metafield来保护metatable,防止用户看到或改变metatable。通过设置metatable中的__metatable
域,getmetatable()
将会返回该域的值,setmetatable()
会报错1
2
3
4mt.__metatable = "not your business"
s1 = Set.new()
print(getmetatable(s1)) --> not your business
setmetatable(s1, {}) --> stdin:1: cannot change protected metatable__index
metamethod用于在索引table中不存在的元素时提供值,如果没有该metamethod,则直接返回nil1
2
3
4
5
6
7
8
9
10
11
12
13-- create the prototype with default values
prototype = {x = 0, y = 0, width = 100, height = 100}
mt = {} -- create a metatable
-- declare the constrctor function
function new(o)
setmetatable(o, mt)
return o
end
mt.__index = function (_, key)
return prototype[key]
end
w = new{x = 10, y = 20}
print(w.width) --> 100
__index
为函数时,两个参数为table和不存在的key。__index
也可以为一个table,此时则在此table中重新查找,查找方式同上,即可以继续用该table的__index
1
mt.__index = prototype
rawget(t, k)
用于对table的raw access,即无视metatable__newindex
用于对table中不存在的元素赋值,可以取代原赋值语句,其可以为函数或table,rawset(t, k, v)
可以进行raw assignment,无视metatable利用
__index
为table设置默认值1
2
3
4
5
6
7
8function setDefault(t, d)
local mt = {__index = function () return d end}
setmetatable(t, mt)
end
tab = {x = 10, y = 20}
print(tab.x, tab.z) --> 10 nil
setDefault(tab, 0)
print(tab.x, tab.z) --> 10 0
为了避免每个table的__index
都生成一个闭包从而浪费资源,可以将默认值存储在table的元素中1
2
3
4
5local mt = {__index = function (t) return t.___ end}
function setDefault(t, d)
t.___ = d
setmetatable(t, mt)
end
为了避免名字冲突,可以用一个局部空table作为索引1
2
3
4
5
6local key = {} -- unique key
local mt = {__index = function (t) return t[key] end}
function setDefault(t, d)
t[key] = d
setmetatable(t, mt)
end
__index
和__newindex
只在索引不存在的元素时才会生效,所以为了监视一个table的所有存取值操作,可以用一个空table作为代理,对这些操作处理之后再重定向到原table上1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19t = {} -- original table (created somewhere)
-- keep a private access to the original table
local _t = t
-- create proxy
t = {}
-- create metatable
local mt = {
__index = function (t, k)
print("*access to element " .. tostring(k))
return _t[k] -- access the original tables
end,
__newindex = function (t, k, v)
print("*update of element " .. tostring(k) .. " to " .. tostring(v))
_t[k] = v -- update original table
end
}
setmetatable(t, mt)
t[2] = "hello" --> *update of element 2 to hello
print(t[2]) --> *access to element 2\nhello__pairs
1
2
3
4mt.__pairs = function ()
return function (_, k)
return next(_t, k)
end
__ipairs
也可以设置
为了避免每个proxy的
__index
和__newindex
都生成一个闭包从而浪费资源,可以将原table存储在proxy的元素中,利用局部空table做索引避免名字冲突1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26local index = {} -- create private index
local mt = { -- create metatable
__index = function (t, k)
print("*access to element " .. tostring(k))
return t[index][k] -- access the original table
end,
__newindex = function (t, k, v)
print("*update of element " .. tostring(k) .. " to " .. tostring(v))
t[index][k] = v -- update original table
end,
__pairs = function (t)
return function (t, k)
return next(t[index], k)
end, t
end
}
function track(t)
local proxy = {}
proxy[index] = t
setmetatable(proxy, mt)
return proxy
end
t = {}
t = track(t)
t[2] = "hello"
print(t[2])只读table
1
2
3
4
5
6
7
8
9
10
11
12
13
14function readOnly(t)
local proxy = {}
local mt = { -- create metatable
__index = t,
__newindex = function (t, k, v)
error("attempt to update a read-only table", 2)
end
}
setmetatable(proxy, mt)
return proxy
end
days = readOnly{"Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"}
print(days[1]) --> Sunday
days[2] = "Noday" --> stdin:1: attempt to update a read-only table