微信公眾號:GameToolDev
關注可了解更多的游戲工具開發教程。問題或建議,請公眾號留言;
從Lua 5.1開始,我們可以使用require和module函數來獲取和創建Lua中的模塊。從使用者的角度來看,一個模塊就是一個程序庫,可以通過require來加載,之后便得到一個類型為table的全局變量。此時的table就像名字空間一樣,可以訪問其中的函數和常量,如:
1?require?"mod"
2?mod.foo()
3?local?m2?=?require?"mod2"
4?local?f?=?mod2.foo
5?f()???
1. require函數:?? ?require函數的調用形式為require "模塊名"。該調用會返回一個由模塊函數組成的table,并且還會定義一個包含該table的全局變量。在使用Lua中的標準庫時可以不用顯示的調用require,因為Lua已經預先加載了他們。?? ?require函數在搜素加載模塊時,有一套自定義的模式,如:?? ??;?.lua;c:/windows/?;/usr/local/lua/?/?.lua?? ?在上面的模式中,只有問號(?)和分號(;)是模式字符,分別表示require函數的參數(模塊名)和模式間的分隔符。如:調用require "sql",將會打開以下的文件:?? ?sql?? ?sql.lua?? ?c:/windows/sql?? ?/usr/local/lua/sql/sql.lua?? ?Lua將require搜索的模式字符串放在變量package.path中。當Lua啟動后,便以環境變量LUA_PATH的值來初始化這個變量。如果沒有找到該環境變量,則使用一個編譯時定義的默認路徑來初始化。如果require無法找到與模塊名相符的Lua文件,就會找C程序庫。C程序庫的搜索模式存放在變量package.cpath中。而這個變量則是通過環境變量LUA_CPATH來初始化的。????? ?2. 編寫模塊的基本方法:?? ?見如下代碼和關鍵性注釋:
1?--將模塊名設置為require的參數,這樣今后重命名模塊時,只需重命名文件名即可。
?2?local?modname?=?...
?3?local?M?=?{}
?4?_G[modname]?=?M
?5?
?6?M.i?=?{r?=?0,?i?=?1}??--定義一個模塊內的常量。
?7?function?M.new(r,i)?return?{r?=?r,?i?=?i}?end
?8?function?M.add(c1,c2)?
?9?????return?M.new(c1.r?+?c2.r,c1.i?+?c2.i)
10?end
11?
12?function?M.sub(c1,c2)
13?????return?M.new(c1.r?-?c2.r,c1.i?-?c2.i)
14?end
15?--返回和模塊對應的table。
16?return?M?
? 3. 使用環境:?? ?仔細閱讀上例中的代碼,我們可以發現一些細節上問題。比如模塊內函數之間的調用仍然要保留模塊名的限定符,如果是私有變量還需要加local關鍵字,同時不能加模塊名限定符。如果需要將私有改為公有,或者反之,都需要一定的修改。那又該如何規避這些問題呢?我們可以通過Lua的函數“全局環境”來有效的解決這些問題。見如下修改的代碼和關鍵性注釋:
1?--模塊設置和初始化。這一點和上例一致。
?2?local?modname?=?...
?3?local?M?=?{}
?4?_G[modname]?=?M
?5?
?6?--聲明這個模塊將會用到的全局函數,因為在setfenv之后將無法再訪問他們,
?7?--因此需要在設置之前先用本地變量獲取。
?8?local?sqrt?=?mat.sqrt
?9?local?io?=?io
10?
11?--在這句話之后就不再需要外部訪問了。
12?setfenv(1,M)
13?
14?--后面的函數和常量定義都無需模塊限定符了。
15?i?=?{r?=?0,?i?=?1}
16?function?new(r,i)?return?{r?=?r,?i?=?i}?end
17?function?add(c1,c2)?
18?????return?new(c1.r?+?c2.r,c1.i?+?c2.i)
19?end
20??
21?function?sub(c1,c2)
22?????return?new(c1.r?-?c2.r,c1.i?-?c2.i)
23?end
24?--返回和模塊對應的table。
25?return?M
? ?4. module函數:?? ?在Lua 5.1中,我們可以用module(...)函數來代替以下代碼,如:
1?local?modname?=?...
2?local?M?=?{}
3?_G[modname]?=?M
4?package.loaded[modname]?=?M
5?????--[[
6 ????和普通Lua程序塊一樣聲明外部函數。
7?????--]]
8?setfenv(1,M)
由于在默認情況下,module不提供外部訪問,必須在調用它之前,為需要訪問的外部函數或模塊聲明適當的局部變量。然后Lua提供了一種更為方便的實現方式,即在調用module函數時,多傳入一個package.seeall的參數,如:?? ?module(...,package.seeall)
版權申明:內容來源網絡,版權歸原創者所有。除非無法確認,我們都會標明作者及出處,如有侵權煩請告知,我們會立即刪除并表示歉意。謝謝。

GameToolDev
ID:GameToolDev
www.GameToolDev.Com
游戲工具開發丨Unity開發技術丨AI技能編輯器丨地圖編輯器