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