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代碼來完成
Lua中的元表(metatable)、元方法(metamethod)詳解