lua会首先把代码编译成中间码然后执行,或许大家都有所困惑:它不是解释性语言吗?其实lua作为解释性语言的意义在于其能在运行过程中完成编译工作,正是dofile、loadfile、require这种函数的存在,lua才能顺理成章地称自己为“解释性语言”。
我们可以通过dofile函数实现在代码中动态加载一个lua文件并执行它:
-- file 'lib1.lua'
function norm (x, y)
local n2 = x^2 + y^2
return math.sqrt(n2)
end
function twice (x)
return 2*x
end
> dofile("lib1.lua") -- load your library
> n = norm(3.4, 1.0)
> print(twice(n)) --> 7.0880180586677
而loadfile和dofile的不同点在于loadfile只是负责加载lua文件而非执行,dofile在loadflie之后执行了一次,而且如若加载失败则第一时间返回错误信息。因此可以这么定义dofile的代码:
function dofile (filename)
local f = assert(loadfile(filename))
return f()
end
loadstring和loadfile类似,只不过接收的是需要执行的字符串而非文件路径。
require和dofile功能类似,但是值得注意的是require有两点不同:
require的路径是一个模式列表,每一个模式指明一种虚文件名到实文件名的一种方法,比如require "some",模式列表为:?;?.lua;c:\windows\?;/usr/local/lua/?/?.lua ,则会从左至右依次搜寻文件直至存在:
some
some.lua
c:\windows\some
/usr/local/lua/some/some.lua.
有一个进阶技巧则是设置模式列表的最后一项为默认lua文件,比如以上模式列表可以是:?;?.lua;c:\windows\?;/usr/local/lua/?/?.lua;default.lua。这么设置的意义即在于未找到对应文件时可以加载默认的lua文件。
Lua自身保存了一个table用以筛查之前加载的文件名。注意这里保存的文件名为虚文件名,因此按照以上模式列表,如果require "some"且require "some.lua",则这个文件会同时加载两遍。
因此require的伪代码可以写作如下形式:
tbMode = {"?","?.lua","c:/windows/?","/usr/local/lua/?/?.lua"}
tbMap = {}
function requireMySelf(strPath)
for i = 1,#tbMap do
if strPath == tbMap[i] then
return
end
end
for i = 1,#tbMode do
if loadfile(string.gsub(tbMode[i],"?","strPath")) then
table.insert(tbMap,strPath)
break
end
end
end
标准C中进行动态库连接是很困难的一件事,但是lua可以通过loadlib的形式来实现动态库链接。通过传入动态链接库的绝对路径和初始化函数即可完成load,如若后续需要执行则需要显示调用以此这个变量即可:
local path = "/usr/local/lua/lib/libluasocket.so"
-- or path = "C:\\windows\\luasocket.dll"
local f = assert(loadlib(path, "luaopen_socket"))
f() -- actually open the library