dofile 當作Lua運行代碼的chunk的一種原始的操作。dofile實際上是一個輔助的函數(shù)。真正完成功能的函數(shù)是loadfile;與dofile不同的是 loadfile編譯代碼成中間碼并且返回編譯后的chunk作為一個函數(shù),而不執(zhí)行代碼;另外loadfile不會拋出錯誤信息而是返回錯誤代。我們可 以這樣定義dofile:
function dofile (filename)
local f = assert(loadfile(filename))
return f()
end
如 果loadfile失敗assert會拋出錯誤。loadfile更加靈活。在發(fā)生錯誤的情況下,loadfile返回nil和錯誤信息,這樣我們就可以 自定義錯誤處理。另外,如果我們運行一個文件多次的話,loadfile只需要編譯一次,但可多次運行。dofile卻每次都要編譯。
2.loadstring與loadfile
loadstring與loadfile相似,只不過它不是從文件里讀入chunk,而是從一個串中讀入。
f = loadstring("i = i + 1")
loadstring 函數(shù)功能強大,但使用時需多加小心。確認沒有其它簡單的解決問題的方法再使用。loadfile和loadstring都不會拋出錯誤,如果發(fā)生錯誤他們 將返回nil加上錯誤信息。另外,loadfile和loadstring都不會有邊界效應(yīng)產(chǎn)生,他們僅僅編譯chunk成為自己內(nèi)部實現(xiàn)的一個匿名函 數(shù)。通常對他們的誤解是他們定義了函數(shù)。Lua中的函數(shù)定義是發(fā)生在運行時的賦值而不是發(fā)生在編譯時。
loadstring通常用 于運行程序外部的代碼,比如運行用戶自定義的代碼。注意:loadstring期望一個chunk,即語句。如果想要加載表達式,需要在表達式前加 return,那樣將返回表達式的值。loadstring返回的函數(shù)和普通函數(shù)一樣,可以多次被調(diào)用。
print "enter your expression:"
local l = io.read()
local func = assert(loadstring("return " .. l))
print("the value of your expression is " .. func())
3.require與dofile
。粗略的說require和dofile完成同樣的功能但有兩點不同:
1. require會搜索目錄加載文件
2. require會判斷是否文件已經(jīng)加載避免重復加載同一文件。由于上述特征,require在Lua中是加載庫的更好的函數(shù)。
require 使用的路徑和普通我們看到的路徑還有些區(qū)別,我們一般見到的路徑都是一個目錄列表。require的路徑是一個模式列表,每一個模式指明一種由虛文件名 (require的參數(shù))轉(zhuǎn)成實文件名的方法。更明確地說,每一個模式是一個包含可選的問號的文件名。匹配的時候Lua會首先將問號用虛文件名替換,然后 看是否有這樣的文件存在。如果不存在繼續(xù)用同樣的方法用第二個模式匹配。例如,路徑如下:?;?.lua;c:\windows\?;/usr/local/lua/?/?.lua
調(diào)用過程如下:
lili
lili.lua
c:\windows\lili
/usr/local/lua/lili/lili.lua
還會有so/dll文件。
為了確定路徑,Lua首先檢查全局變量LUA_PATH是否為一個字符串,如果是則認為這個串就是路徑;否則require檢查環(huán)境變量LUA_PATH的值,如果兩個都失敗require使用固定的路徑(典型的"?;?.lua")
一個路徑中的模式也可以不包含問號而只是一個固定的路徑,比如:?;?.lua;/usr/local/default.lua。這種情況下,require沒有匹配的時候就會使用這個固定的文件(當然這個固定的路徑必須放在模式列表的最后才有意義)