亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Lua中的__index和__newindex實(shí)例

 更新時(shí)間:2014年09月29日 10:37:15   作者:果凍想  
這篇文章主要介紹了Lua中的__index和__newindex實(shí)例,本文講解了具有默認(rèn)值的table、記錄table的訪問、只讀的table等內(nèi)容,需要的朋友可以參考下

前言

這篇博文將通過幾個(gè)簡(jiǎn)單的實(shí)例演示,鞏固對(duì)__index和__newindex的理解,同時(shí)加深對(duì)Lua中元表和元方法的理解,如果對(duì)Lua的元表和元方法還不是很熟悉的話,請(qǐng)參考這篇文章:《Lua中的元表與元方法》。

具有默認(rèn)值的table

我們都知道,table中的任何字段的默認(rèn)值都是nil,但是通過元表,我們可以很容易的修改這一規(guī)定,代碼如下:

復(fù)制代碼 代碼如下:

function setDefault(tb, defaultValue)
     local mt = {__index = function () return defaultValue end}
     setmetatable(tb, mt)
end
 
local tb1 = {x = 10, y = 20}
print(tb1.x, tb1.z)     --> 10 nil
setDefault(tb1, 100) -->設(shè)置默認(rèn)值
print(tb1.x, tb1.z) --> 10 100 這里打印的就是默認(rèn)值

可以看到,在代碼中,setDefault函數(shù)為所有需要默認(rèn)值的table創(chuàng)建了一個(gè)新的元表。如果準(zhǔn)備創(chuàng)建很多需要默認(rèn)值得table,這種方法的開銷或許就比較大了。由于在元表中默認(rèn)值defaultValue是與元方法關(guān)聯(lián)在一起的,所以setDefault無法為所有table都使用同一個(gè)元表。如果要讓具有不同默認(rèn)值得table都使用同一個(gè)元表,那么就需要將每個(gè)元表的默認(rèn)值存放在table本身中,可以使用一個(gè)額外的字段來存儲(chǔ)默認(rèn)值。例如以下代碼:

復(fù)制代碼 代碼如下:

local mt = {__index = function (t) return t.___ end}
function setDefault(tb, defaultValue)
     tb.___ = defaultValue       -- 非常謝謝hellowei犀利的review。具體請(qǐng)參見評(píng)論
     setmetatable(tb, mt)
end

上面代碼中的“___”是為了防止名字沖突而起的名字;如果這樣的話,你還擔(dān)心名字沖突,確保key在table中的唯一性,只需要?jiǎng)?chuàng)建一個(gè)新的table,并用它作為key即可,每一個(gè)新創(chuàng)建的table都是一個(gè)唯一的地址,比如以下代碼:

復(fù)制代碼 代碼如下:

local key = {} -- 唯一的key
local mt = {__index = function (tb) return tb[key] end}
 
function setDefault(tb, defaultValue)
     tb[key] = defaultValue
     setmetatable(tb, mt)
end

記錄table的訪問

有的時(shí)候,一種特定的需求,我們需要記錄對(duì)一個(gè)table的所有訪問,不管是查詢還是更新,我們都需要記錄日志。這如何完成?我們都知道,元表中的__index和__newindex是在table中沒有所需要訪問的index時(shí)才發(fā)揮作用的,因此,只有將一個(gè)table保持為空,然后設(shè)置__index和__newindex元方法,才有可能記錄下來所有對(duì)它的訪問。

為了監(jiān)視一個(gè)table的所有訪問,就應(yīng)該為真正的table創(chuàng)建一個(gè)代理。這個(gè)代理就是一個(gè)空的table,其中__index和__newindex元方法可用于跟蹤所有的訪問,并將訪問重定義到原來的table上。這就是思路,接下來看代碼:

復(fù)制代碼 代碼如下:

local t = {} --原來的table
 
-- 保持對(duì)原table的一個(gè)引用
local _t = t
 
-- 創(chuàng)建代理
t = {}
 
-- 創(chuàng)建元表
local mt = {
__index = function (t, k)
print("access to element " .. tostring(k))
return _t[k]
end,
 
__newindex = function (t, k, v)
print("update of element " .. tostring(k))
_t[k] = v
end
}
 
setmetatable(t, mt)
 
t.x = 10 -- update of element x
print(t.x) -- access to element x

如果想要同時(shí)監(jiān)視幾個(gè)table,無須為每個(gè)table創(chuàng)建不同的元表;相反,只要以某種形式將每個(gè)代理與其原table關(guān)聯(lián)起來,并且所有代理都共享一個(gè)公共的元表。這個(gè)問題與設(shè)置table默認(rèn)值相關(guān)聯(lián)的問題類似,也是將原來的table保存在代理table的一個(gè)特殊的字段中。代碼如下:

復(fù)制代碼 代碼如下:

-- 創(chuàng)建唯一索引
local index = {}
 
-- 創(chuàng)建元表
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))
          t[index][k] = v
     end
}
 
function track(t)
     local proxy = {}
     proxy[index] = t
     setmetatable(proxy, mt)
     return proxy
end
 
local t = {}
local proxy = track(t)
proxy.x = 10
print(proxy.x)

只讀的table

通過代理的概念,可以很容易的實(shí)現(xiàn)只讀的table。只需要跟蹤所有對(duì)table的更新操作,并引發(fā)一個(gè)錯(cuò)誤就好了,對(duì)于查詢時(shí),我們不用去館,只需要管對(duì)table的更新操作,廢話不說,來段簡(jiǎn)單的代碼,自然而然的一目了然了。

復(fù)制代碼 代碼如下:

function readOnly(t)
     local proxy = {}
 
     -- 創(chuàng)建元表
     local mt = {
          __index = t,
          __newindex = function (t, k, v)
               error("Attempt to update a read-only table", 2)
          end
     }
 
     setmetatable(proxy, mt)
     return proxy
end
 
local tbDemo = readOnly{1, 2, 3, 4, 5}
print(tbDemo[1])
tbDemo[1] = 20

元表中__index對(duì)應(yīng)的是原來的table,而更新原來的table時(shí),就會(huì)顯示錯(cuò)誤提示:Attempt to update a read-only table。

總結(jié)

這篇文章對(duì)Lua中的__index和__newindex的使用進(jìn)行了詳細(xì)的講解和分析,并提供了實(shí)際的代碼,主要是為了加深對(duì)Lua中元表和元方法的理解,元表和元方法在Lua中的地位太總要了,很多高級(jí)的編程技巧和特殊需求都是基于元表和元方法來實(shí)現(xiàn)了,所以,也希望大家能好好的閱讀這篇文章,同時(shí)也希望我的文章對(duì)大家有幫助。

相關(guān)文章

  • Lua中的運(yùn)算符簡(jiǎn)明總結(jié)

    Lua中的運(yùn)算符簡(jiǎn)明總結(jié)

    這篇文章主要介紹了Lua中的運(yùn)算符簡(jiǎn)明總結(jié),本文總結(jié)了數(shù)學(xué)運(yùn)算符、連接運(yùn)算、賦值運(yùn)算、 邏輯運(yùn)算、關(guān)系運(yùn)算等運(yùn)算符,需要的朋友可以參考下
    2014-10-10
  • Lua math.fmod使用時(shí)的小數(shù)問題

    Lua math.fmod使用時(shí)的小數(shù)問題

    這篇文章主要介紹了Lua math.fmod使用時(shí)的小數(shù)問題,math.fmod用于取模運(yùn)算,使用小數(shù)時(shí)可能會(huì)遇到不可預(yù)料的結(jié)果,所以應(yīng)該避免使用小數(shù),需要的朋友可以參考下
    2015-06-06
  • Lua中實(shí)現(xiàn)php的strpos()以及strrpos()函數(shù)

    Lua中實(shí)現(xiàn)php的strpos()以及strrpos()函數(shù)

    這篇文章主要介紹了在Lua中實(shí)現(xiàn)php的strpos()以及strrpos()函數(shù)的方法,需要的朋友可以參考下
    2014-11-11
  • Lua教程(十三):弱引用table

    Lua教程(十三):弱引用table

    這篇文章主要介紹了Lua教程(十三):弱引用table,一個(gè)table的弱引用類型是通過其元表的__mode字段來決定的,如果該值為包含字符"k",那么table就是key弱引用,如果包含"v",則是value弱引用,如果兩個(gè)字符均存在,就是key/value弱引用,需要的朋友可以參考下
    2015-04-04
  • Lua中的模塊與module函數(shù)詳解

    Lua中的模塊與module函數(shù)詳解

    這篇文章主要介紹了Lua中的模塊與module函數(shù)詳解,本文講解了編寫一個(gè)簡(jiǎn)單的模塊、避免修改模塊名的方法、模塊名參數(shù)等內(nèi)容,需要的朋友可以參考下
    2014-09-09
  • Lua中break語句的使用方法詳解

    Lua中break語句的使用方法詳解

    這篇文章主要介紹了Lua中break語句的使用方法詳解,是Lua入門學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下
    2015-05-05
  • lua腳本實(shí)現(xiàn)自動(dòng)生成APK包

    lua腳本實(shí)現(xiàn)自動(dòng)生成APK包

    上次用了純bat寫了個(gè)腳本生成APK包,感覺bat擴(kuò)展性和語法差的令人發(fā)指,這次用lua重寫了一個(gè)腳本,適用于cocos2dx+lua的項(xiàng)目,小伙伴們來參考下吧。
    2015-03-03
  • Lua的迭代器使用中應(yīng)該避免的問題和技巧

    Lua的迭代器使用中應(yīng)該避免的問題和技巧

    這篇文章主要介紹了Lua的迭代器使用中應(yīng)該避免的問題和技巧,本文介紹了避免創(chuàng)建閉合函數(shù)、利用恒定狀態(tài)創(chuàng)造更多變量、不需要for循環(huán)的迭代器等內(nèi)容,需要的朋友可以參考下
    2014-09-09
  • Lua教程(六):綁定一個(gè)簡(jiǎn)單的C++類

    Lua教程(六):綁定一個(gè)簡(jiǎn)單的C++類

    這篇文章主要介紹了Lua教程(六):綁定一個(gè)簡(jiǎn)單的C++類,本文是最后一篇C/C++與Lua交互的教程,其他教程請(qǐng)參閱本文下方的相關(guān)文章,需要的朋友可以參考下
    2014-09-09
  • 簡(jiǎn)單講解Lua中的垃圾回收機(jī)制

    簡(jiǎn)單講解Lua中的垃圾回收機(jī)制

    這篇文章主要介紹了Lua中的垃圾回收機(jī)制,自動(dòng)的垃圾回收是Lua的重要特性之一,需要的朋友可以參考下
    2015-05-05

最新評(píng)論