佳木斯湛栽影视文化发展公司

主頁 > 知識庫 > Lua進(jìn)階教程之閉包函數(shù)、元表實例介紹

Lua進(jìn)階教程之閉包函數(shù)、元表實例介紹

熱門標(biāo)簽:百度競價排名 呼叫中心市場需求 服務(wù)外包 鐵路電話系統(tǒng) Linux服務(wù)器 地方門戶網(wǎng)站 網(wǎng)站排名優(yōu)化 AI電銷

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

function createCountdownTimer(second)
   local ms=second * 1000;
   local function countDown()
      ms = ms - 1;
   return ms;
 end
 return countDown;
end

timer1 = createCountdownTimer(1);
for i=1,3 do
   print(timer1());
end
print("------------");
timer2 = createCountdownTimer(1);
for i=0,2 do
   print(timer2());
end


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

999
998
997
------------
999
998
997

Upvalue:一個函數(shù)所使用的定義在它的函數(shù)體之外的局部變量(external localvariable)稱為這個函數(shù)的upvalue。

在前面的代碼中,函數(shù)countDown使用的定義在函數(shù)createCountdownTimer中的局部變量ms就是countDown的upvalue,但ms對createCountdownTimer而言只是一個局部變量,不是upvalue。Upvalue是Lua不同于C/C++的特有屬性,需要結(jié)合代碼仔細(xì)體會。

函數(shù)閉包:一個函數(shù)和它所使用的所有upvalue構(gòu)成了一個函數(shù)閉包。

閉包是一個內(nèi)部函數(shù),它可以訪問一個或者多個外部函數(shù)的外部局部變量。每次閉包的成功調(diào)用后這些外部局部變量都保存他們的值(狀態(tài))。當(dāng)然如果要創(chuàng)建一個閉包必須要創(chuàng)建其外部局部變量。所以一個典型的閉包的結(jié)構(gòu)包含兩個函數(shù):一個是閉包自己;另一個是工廠(創(chuàng)建閉包的函數(shù))。迭代器需要保留上一次成功調(diào)用的狀態(tài)和下一次成功調(diào)用的狀態(tài),也就是他知道來自于哪里和將要前往哪里。閉包提供的機(jī)制可以很容易實現(xiàn)這個任務(wù)。

Lua函數(shù)閉包與C函數(shù)的比較:Lua函數(shù)閉包使函數(shù)具有保持它自己的狀態(tài)的能力,從這個意義上說,可以與帶靜態(tài)局部變量的C函數(shù)相類比。但二者有顯著的不同:對Lua來說,函數(shù)是一種基本數(shù)據(jù)類型——代表一種(可執(zhí)行)對象,可以有自己的狀態(tài);但是對帶靜態(tài)局部變量的C函數(shù)來說,它并不是C的一種數(shù)據(jù)類型,更不會產(chǎn)生什么對象實例,它只是一個靜態(tài)地址的符號名稱。

基于對象的實現(xiàn)方式

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

function create(name,id)
    local data={name = name,id=id};
    local obj={};
    function obj.GetName()
      return data.name;
 end
 function obj.GetID()
    return data.id;
 end
 function obj.SetName(name)
    data.name=name;
 end
 function obj.SetID(id)
    data.id=id
 end
 return obj;
end

o1 = create("Sam", 001)
o2 = create("Bob", 007)
o1.SetID(100)
print("o1's id:", o1.GetID(), "o2's id:",o2.GetID())
o2.SetName("Lucy")
print("o1's name:", o1.GetName(),"o2's name:", o2.GetName())

--o1's id: 100 o2's id: 7
--o1's name: Sam o2's name: Lucy

實現(xiàn)方式:把需要隱藏的成員放在一張表里,把該表作為成員函數(shù)的upvalue。
局限性:基于對象的實現(xiàn)不涉及繼承及多態(tài)。但另一方面,腳本編程是否需要繼承和多態(tài)要視情況而定。

元表

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

t = {}
m = { a = " and ", b = "Li Lei", c = "Han Meimei" }
setmetatable(t, { __index = m}) --表{ __index=m }作為表t的元表
for k, v in pairs(t) do --窮舉表t
    print(v)
end
print("-------------")
print(t.b, t.a, t.c)

--輸出結(jié)果
---------------
--Li Lei  and  Han Meimei

function add(t1, t2)
    --‘#'運算符取表長度
    assert(#t1 == #t2)
    local length = #t1
    for i = 1, length do
    t1[i] = t1[i] + t2[i]
    end
    return t1
end
--setmetatable返回被設(shè)置的表
t1 = setmetatable({ 1, 2, 3}, { __add = add })
t2 = setmetatable({ 10, 20, 30 }, {__add = add })

t1 = t1 + t2
for i = 1, #t1 do
    print(t1[i])
end
--11
--22
--33

定義:元表本身只是一個普通的表,通過特定的方法(比如setmetatable)設(shè)置到某個對象上,進(jìn)而影響這個對象的行為;一個對象有哪些行為受到元表影響以及這些行為按照何種方式受到影響是受Lua語言約束的。比如在前面的代碼里,兩個表對象的加法運算,如果沒有元表的干預(yù),就是一種錯誤;但是Lua規(guī)定了元表可以“重載”對象的加法運算符,因此若把定義了加法運算的元表設(shè)置到那兩個表上,它們就可以做加法了。元表是Lua最關(guān)鍵的概念之一,內(nèi)容也很豐富,請參考Lua文檔了解詳情。

元表與C++虛表的比較:如果把表比作對象,元表就是可以改變對象行為的“元”對象。在某種程度上,元表可以與C++的虛表做一類比。但二者還是迥然不同的:元表可以動態(tài)的改變,C++虛表是靜態(tài)不變的;元表可以影響表(以及其他類型的對象)的很多方面的行為,虛表主要是為了定位對象的虛方法(最多再帶上一點點RTTI)。

您可能感興趣的文章:
  • Lua學(xué)習(xí)筆記之表和函數(shù)
  • Lua基礎(chǔ)教程之賦值語句、表達(dá)式、流程控制、函數(shù)學(xué)習(xí)筆記
  • Lua中的閉包小結(jié)
  • Lua中的閉包學(xué)習(xí)筆記
  • LUA中的閉包(closure)淺析
  • Lua學(xué)習(xí)筆記之函數(shù)、變長參數(shù)、closure(閉包)、select等
  • lua閉包的理解以及表與函數(shù)的幾種表達(dá)方法

標(biāo)簽:湘潭 仙桃 銅川 蘭州 湖南 衡水 崇左 黃山

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Lua進(jìn)階教程之閉包函數(shù)、元表實例介紹》,本文關(guān)鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    新丰县| 宁海县| 天镇县| 涿鹿县| 长阳| 慈溪市| 黔西| 多伦县| 平凉市| 永吉县| 游戏| 凤台县| 前郭尔| 永修县| 吉首市| 广灵县| 三都| 景洪市| 深泽县| 海伦市| 景东| 启东市| 五莲县| 龙口市| 盐山县| 永清县| 农安县| 台江县| 西藏| 苗栗市| 吉安县| 陆良县| 修武县| 界首市| 延川县| 澳门| 奎屯市| 西乌珠穆沁旗| 丘北县| 晋城| 莱州市|