Lua查找表元素過(guò)程(元表、__index方法是如何工作的)

轉(zhuǎn)自 http://blog.csdn.net/xocoder/article/details/9028347

近日開(kāi)始頻繁使用Lua,于是把這塊東西理了一下,特此記錄。Lua的表本質(zhì)其實(shí)是個(gè)類(lèi)似HashMap的東西,其元素是很多的Key-Value對(duì),如果嘗試訪問(wèn)了一個(gè)表中并不存在的元素時(shí),就會(huì)**觸發(fā)Lua的一套查找機(jī)制,也是憑借這個(gè)機(jī)制來(lái)模擬了類(lèi)似“類(lèi)”的行為

tempTable = {}
print(tempTable.memberA) --這里試圖打印tempTable并不存在的成員memberA

執(zhí)行結(jié)果:nil輸出為nil的原因很簡(jiǎn)單,tempTable中并沒(méi)有memberA這個(gè)成員,這符合我們平時(shí)對(duì)HashMap的認(rèn)知。但對(duì)于Lua表,如果tempTable有元表,情況就不同了。

什么是元表:元表像是一個(gè)“操作指南”,里面包含了一系列操作的解決方案,例如__index方法就是定義了這個(gè)表在索引失敗的情況下該怎么辦。

__index元方法:按照之前的說(shuō)法,如果A的元表是B,那么如果訪問(wèn)了一個(gè)A中不存在的成員,就會(huì)訪問(wèn)查找B中有沒(méi)有這個(gè)成員。這個(gè)過(guò)程大體是這樣,但卻不完全是這樣,實(shí)際上,即使將A的元表設(shè)置為B,而且B中也確實(shí)有這個(gè)成員,返回結(jié)果仍然會(huì)是nil,原因就是B的__index元方法沒(méi)有賦值。別忘了我們之前說(shuō)過(guò)的:“元表是一個(gè)操作指南”,定義了元表,只是有了操作指南,但不應(yīng)該在操作指南里面去查找元素,而__index方法則是“操作指南”的“索引失敗時(shí)該怎么辦”。這么說(shuō)有點(diǎn)繞。所以:

father = {
house=1
}
son = {
car=1
}
setmetatable(son, father) --把son的metatable設(shè)置為father
print(son.house)

輸出的結(jié)果是nil,但如果把代碼改為

father = {
house=1
}
father.__index = father -- 把father的__index方法指向自己
son = {
car=1
}
setmetatable(son, father)
print(son.house)

輸出的結(jié)果為1,符合預(yù)期

這樣一來(lái),結(jié)合上例,來(lái)解釋__index元方法的含義:
在上述例子中,訪問(wèn)son.house時(shí),son中沒(méi)有house這個(gè)成員,但Lua接著發(fā)現(xiàn)son有元表father,注意:此時(shí),Lua并不是直接在father中找名為house的成員,而是調(diào)用father的__index方法,如果__index方法為nil,則返回nil,如果是一個(gè)表(上例中father的__index方法等于自己,就是這種情況),那么就到__index方法所指的這個(gè)表中查找名為house的成員,于是,最終找到了house成員。注:__index方法除了可以是一個(gè)表,還可以是一個(gè)函數(shù),如果是一個(gè)函數(shù),__index方法被調(diào)用時(shí)將返回該函數(shù)的返回值。

到這里,總結(jié)一下Lua查找一個(gè)表元素時(shí)的規(guī)則,其實(shí)就是如下3個(gè)步驟:
1.在表中查找,如果找到,返回該元素,找不到則繼續(xù)
2.判斷該表是否有元表(操作指南),如果沒(méi)有元表,返回nil,有元表則繼續(xù)
3.判斷元表有沒(méi)有__index方法,如果__index方法為nil,則返回nil;如果__index方法是一個(gè)表,則重復(fù)1、2、3;如果__index方法是一個(gè)函數(shù),則返回該函數(shù)的返回值

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容

  • Nginx API for Lua Introduction ngx.arg ngx.var.VARIABLE C...
    吃瓜的東閱讀 5,869評(píng)論 0 5
  • 第一篇 語(yǔ)言 第0章 序言 Lua僅讓你用少量的代碼解決關(guān)鍵問(wèn)題。 Lua所提供的機(jī)制是C不擅長(zhǎng)的:高級(jí)語(yǔ)言,動(dòng)態(tài)...
    testfor閱讀 2,719評(píng)論 1 7
  • 前言 元表對(duì)應(yīng)的英文是metatable,元方法是metamethod。我們都知道,在C++中,兩個(gè)類(lèi)是無(wú)法直接相...
    BobWong閱讀 1,059評(píng)論 0 9
  • 學(xué)習(xí)lua也有大概一年了,對(duì)lua的一些基本的語(yǔ)法很熟練了,也做了一些簡(jiǎn)單的業(yè)務(wù),但是對(duì)于lua的高級(jí)特性還是不是...
    ytlm閱讀 1,915評(píng)論 0 2
  • 把歌聲帶回家 讓潺潺的流水撫摸睡夢(mèng)中的他 奔波的生活 你勞累啦 把歌聲帶回家 讓輕輕的風(fēng)兒親吻夢(mèng)里的他 方向盤(pán)很圓...
    四月麗人閱讀 215評(píng)論 0 2