從Lua 5.1開始,我們可以使用require和module函數(shù)來獲取和創(chuàng)建Lua中的模塊。從使用者的角度來看,一個(gè)模塊就是一個(gè)程序庫,可以通過require來加載,之后便得到一個(gè)類型為table的全局變量。此時(shí)的table就像名字空間一樣,可以訪問其中的函數(shù)和常量,如:
1. require函數(shù):
require函數(shù)的調(diào)用形式為require "模塊名"。該調(diào)用會(huì)返回一個(gè)由模塊函數(shù)組成的table,并且還會(huì)定義一個(gè)包含該table的全局變量。在使用Lua中的標(biāo)準(zhǔn)庫時(shí)可以不用顯示的調(diào)用require,因?yàn)長ua已經(jīng)預(yù)先加載了他們。
require函數(shù)在搜素加載模塊時(shí),有一套自定義的模式,如:
見如下代碼和關(guān)鍵性注釋:
M.i = {r = 0, i = 1} --定義一個(gè)模塊內(nèi)的常量。
function M.new(r,i) return {r = r, i = i} end
function M.add(c1,c2)
return M.new(c1.r + c2.r,c1.i + c2.i)
end
function M.sub(c1,c2)
return M.new(c1.r - c2.r,c1.i - c2.i)
end
--返回和模塊對(duì)應(yīng)的table。
return M
3. 使用環(huán)境:
仔細(xì)閱讀上例中的代碼,我們可以發(fā)現(xiàn)一些細(xì)節(jié)上問題。比如模塊內(nèi)函數(shù)之間的調(diào)用仍然要保留模塊名的限定符,如果是私有變量還需要加local關(guān)鍵字,同時(shí)不能加模塊名限定符。如果需要將私有改為公有,或者反之,都需要一定的修改。那又該如何規(guī)避這些問題呢?我們可以通過Lua的函數(shù)“全局環(huán)境”來有效的解決這些問題。見如下修改的代碼和關(guān)鍵性注釋:
--聲明這個(gè)模塊將會(huì)用到的全局函數(shù),因?yàn)樵趕etfenv之后將無法再訪問他們,
--因此需要在設(shè)置之前先用本地變量獲取。
local sqrt = mat.sqrt
local io = io
--在這句話之后就不再需要外部訪問了。
setfenv(1,M)
--后面的函數(shù)和常量定義都無需模塊限定符了。
i = {r = 0, i = 1}
function new(r,i) return {r = r, i = i} end
function add(c1,c2)
return new(c1.r + c2.r,c1.i + c2.i)
end
function sub(c1,c2)
return new(c1.r - c2.r,c1.i - c2.i)
end
--返回和模塊對(duì)應(yīng)的table。
return M
4. module函數(shù):
在Lua 5.1中,我們可以用module(...)函數(shù)來代替以下代碼,如:
由于在默認(rèn)情況下,module不提供外部訪問,必須在調(diào)用它之前,為需要訪問的外部函數(shù)或模塊聲明適當(dāng)?shù)木植孔兞俊H缓驦ua提供了一種更為方便的實(shí)現(xiàn)方式,即在調(diào)用module函數(shù)時(shí),多傳入一個(gè)package.seeall的參數(shù),如:
注意:5.2已經(jīng)不支持module了,去看看lua官網(wǎng)的文檔,沒有這個(gè)函數(shù)了,我用package.loaded.module_name = _ENV來創(chuàng)建模塊
新聞熱點(diǎn)
疑難解答
圖片精選