上一篇我們簡單地介紹了Lua的函數(shù),這次,我們來點特別的,來介紹一下Lua的函數(shù)(小若:等等,我是不是錯過了什么?)
1.閉合函數(shù)(closure)
理論上來說,Lua的所有函數(shù)都應(yīng)該稱之為閉合函數(shù),但是,這種反人類的做法,我們還是拋棄吧~
按書上的描述,一個閉合函數(shù)就是:一個函數(shù)加上該函數(shù)所需訪問的所有”非局部的變量“。
理論什么的,很煩人,來看看一個函數(shù):
復(fù)制代碼 代碼如下:
function count()
local i = 0;
return function()
i = i + 1;
return i;
end
end
這個count函數(shù)會返回另外一個函數(shù),重點是,這個返回的函數(shù)會使用count函數(shù)的局部變量。
先來運(yùn)行,看看效果,使用如下方式調(diào)用:
復(fù)制代碼 代碼如下:
local func = count();
print(func());
print(func());
print(func());
輸出結(jié)果如下:
復(fù)制代碼 代碼如下:
[LUA-print] 1
[LUA-print] 2
[LUA-print] 3
怎么旁白不出來咆哮一下?(小若:…咳咳…為毛線會這樣?!不應(yīng)該全部都輸出1嗎??。?br />
在這里,local i就屬于一個非局部變量,因為它既不是全局變量,也不是單純的局部變量(因為另外一個函數(shù)可以反問到它)。
再來回到定義,count函數(shù)里的那個函數(shù),加上非局部變量i,就構(gòu)成了一個閉合函數(shù)了,就這么簡單。
對于閉合函數(shù)而已,屬于它的非局部變量,并不是在調(diào)用它的時候臨時產(chǎn)生的,而是和它一起存在的。
所以每次調(diào)用閉合函數(shù),非局部變量的值都不會被重置。
如果大家還是不太清楚,那么,我們給這個閉合函數(shù)添加一個局部變量吧,修改count函數(shù)如下:
復(fù)制代碼 代碼如下:
function count()
return function()
local i = 999;
i = i + 1;
return i;
end
end
這次,把i作為這個內(nèi)部函數(shù)的局部變量了,它不再是“非局部變量”。
仍然像這樣調(diào)用:
復(fù)制代碼 代碼如下:
local func = count();
print(func());
print(func());
print(func());
輸出結(jié)果如下:
復(fù)制代碼 代碼如下:
[LUA-print] 1000
[LUA-print] 1000
[LUA-print] 1000
陷入, 每一次i變量的值都是全新的。
閉合函數(shù)的用處可大著了,我們在開發(fā)過程中使用的頻率應(yīng)該還算比較大的~
2.非全局函數(shù)
又是這種看著就反人類的名詞,非全局,那就是說,不是全局的函數(shù)(小若:廢你個話?。。?/p>
來看看這樣的一個函數(shù):
復(fù)制代碼 代碼如下:
local function mutou()
end
這就是一個非全局函數(shù),但,這么簡單的東西我可不會拿出來說~
這里我想介紹一個“語法糖”,上面的mutou函數(shù),其實相當(dāng)于以下的代碼:
復(fù)制代碼 代碼如下:
local mutou;
mutou = function ()
end
這就是我們函數(shù)的真實面貌,函數(shù)名稱其實也是一個變量名而已。
所以,有時候,我們在定義函數(shù)的時候,要注意一下順序。比如,這樣的兩個函數(shù):
復(fù)制代碼 代碼如下:
local function mutou()
print("mutou");
return pangbai();
end
local function pangbai()
print("pangbai");
end
mutou函數(shù)里要調(diào)用pangbai函數(shù)。
在編譯的時候,mutou函數(shù)是編譯不過的,因為這個時候pangbai函數(shù)未定義,換句話說,pangbai變量并不存在。
只要換一個方式就可以解決這個問題了:
復(fù)制代碼 代碼如下:
local mutou;
local pangbai;
mutou = function ()
print("mutou");
return pangbai();
end
pangbai = function ()
print("pangbai");
end
這樣編譯就沒問題了,好吧,也許Lua里不該叫編譯,反正,不會報語法錯誤了~
3.尾調(diào)用
好了,一個名詞比一個名詞要反人類。
尾調(diào)用的大致意思是:一個函數(shù)的調(diào)用是另一個函數(shù)的最后一個動作時,這個調(diào)用就稱之為尾調(diào)用。
比如這樣的函數(shù):
復(fù)制代碼 代碼如下:
function mutou()
return count();
end
當(dāng)mutou函數(shù)調(diào)用完count函數(shù)之后,就沒有其他事情要做了,所以,調(diào)用count函數(shù),就屬于尾調(diào)用。
但,如果是這樣的函數(shù):
復(fù)制代碼 代碼如下:
function mutou()
return count() + 1;
end
這就不屬于尾調(diào)用,因為調(diào)用完count函數(shù)之后,還要取得count的返回值,然后進(jìn)行一次加法操作,這就不符合定義了。
尾調(diào)用有什么意義呢?
進(jìn)行尾調(diào)用時不會耗費多余的棧空間,比如這樣一個經(jīng)典的函數(shù):
復(fù)制代碼 代碼如下:
function foo(n)
if n > 0 then
return foo(n - 1);
else
return "end";
end
end
我們可以試試這樣調(diào)用:
復(fù)制代碼 代碼如下:
print(foo(99999));
結(jié)果就是不出所料地輸出了end字符串。
當(dāng)n > 0時,函數(shù)就直接返回foo(n – 1),接著就沒有后續(xù)的動作了,所以這符合尾調(diào)用的定義。
因此,這個函數(shù)的調(diào)用不會引起棧溢出。
但,如果稍微改改,變成這樣:
復(fù)制代碼 代碼如下:
function foo(n)
if n > 0 then
return foo(n - 1) + 0;
else
return "end";
end
end
運(yùn)行的時候,就會報這樣的一個錯誤: [string "src/main.lua"]:57: stack overflow
由于我用Lua的情況不多,所以暫時沒有舉個比較實用的例子,但,這自然會是一個很好用的特性。
好了,關(guān)于Lua的函數(shù),應(yīng)該都介紹完了,我也是按著書的順序在過一遍基礎(chǔ),然后把覺得有意思的部分用文章記錄下來。
您可能感興趣的文章:- js尾調(diào)用優(yōu)化的實現(xiàn)
- 深入理解JavaScript中的尾調(diào)用(Tail Call)
- es6函數(shù)之尾遞歸用法實例分析
- es6函數(shù)name屬性功能與用法實例分析
- ES6學(xué)習(xí)筆記之字符串、數(shù)組、對象、函數(shù)新增知識點實例分析
- ES6中箭頭函數(shù)的定義與調(diào)用方式詳解
- ES6中的箭頭函數(shù)實例詳解
- ES6記錄異步函數(shù)的執(zhí)行時間詳解
- ES6新特性之函數(shù)的擴(kuò)展實例詳解
- es6函數(shù)之尾調(diào)用優(yōu)化實例分析