一. 编译,执行与错误
1. 编译:
1). loadfile会从一个文件加载Lua代码块,但它不会运行代码,只是编译代码,然后将编译结果作为一个函数返回,此外,与dofile不同的还有loadfile不会引发错误,它只是返回错误值并不处理错误;
2). dofile: 加载并且运行lua文件
function dofile(filename)
local f = assert(loadfile(filename))
return f()
end
3). loadstring: 与loadfile类似,不同之处在于它是从一个字符串中读取代码,而非从文件读取。
f = loadsting("i = i +1") 基本等价于 f = function () i = i + 1 end。但是由于loadstring在编译时并不涉及词法域(lexical scoping),上面的代码并不等价。 loadstirng最典型的用途是执行外部代码,那些位于程序之外的代码。值得注意的是,loadstring的期望输入是一个程序块,也就是一系列的语句,如果需要对一个表达式求值,则必须在其之前添加return,这样才能构成一条语句,返回表达式的值。
4). loadfile与loadstring其实在lua中有一个真正的原始函数load,
5). Lua将所有独立的程序块视为一个匿名函数的函数体,并且该匿名函数还具有可变长实参。在内部使用local x = …来获取可变实参
2. 错误:对于非常通用代码,lua提供了内建的函数assert
if not <condition> then error end 等价于 assert( <condition>)
3. 错误处理与异常
对于大多数应用程序而言,无须在Lua代码中作任何错误处理,应用程序本身会负责这类问题。如果需要在lua中处理代码,则必须使用函数pcall来包装需要执行的代码;pcall函数会以一种"保护模式"来调用它的第一个参数(函数或匿名函数),因此pcall可以捕获函数执行中的任何错误;如果没有发生错误会返回true或者函数调用的返回值,否则,返回false以及错误消息;
任何类型的lua值都可以作为错误消息传递给error函数,并且这些值会成为pcall的返回值
4. 错误消息与追溯(traceback)
pcall返回其错误消息时,它已经销毁了调用栈的部分内容,因此,Lua提供了函数xpcall,该函数除了接受一个需要被调用的函数之外,还接受一个错误处理函数。当发生错误时,Lua会在调用栈展开前调用函数错误处理函数,于是就可以使用debug库来获取关于错误的额外信息。
5. debug库提供了两个通用的错误处理函数,一个debug.debug,提供了一个lua提示符,让用户来检查错误的原因;另一个是debug.traceback,根据调用栈来构建一个扩展的错误消息。解释器也是用debug.traceback来构建其错误消息的。