在Lua中,对于大多数程序都不用作任何错误处理,应用程序本身会负责这类问题。

所有的Lua活动都是由应用程序的一次调用开始的,这类调用要求Lua执行一个程序块。

执行过程中发生了错误,此调用会返回一个错误代码(nil ),这样应用程序就能采取适当的行动来处理。

如果需要在Lua中处理错误,则必须使用函数pcall来包装需要执行的代码。

假设在执行一段Lua代码时,捕获所有执行中引发的错误,那么第一步就是将这段代码封装到一个匿名函数中

然后用pcall去调用这个函数:

local ok,msg = pcall(function ()
    <some code>
    if unexcepted then error() end
    <some code>
    print(a[i])    --引发错误,a可能不是一个talbe
    <some code >
end)

if ok then -- 没有错误发生
    <regular code >
else
    <error handling code > --保护模式代码引发一个错误,需要app引起注意
end



如果function运行过程中没有发生错误,pcall返回true,和一些返回值。

否则返回false,和错误消息。

传给error函数的错误消息不一定非得是字符串,它可以是任意的lua值 。这些值也会成为pcall的返回值。



local status,err = pcall( function() error({ code = 121}) end ) --传给error的是table
print(err.code)    --> 121



有了这些机制,就可以完成所有的异常处理。通过使用error来抛出一个异常或使用pcall来捕获异常。错误消息

则可以标识出错误类型或内容。

虽然错误消息可以是任意值,但是通常使用字符串去描述什么样的错误。

当发生内部错误时,Lua就抛出一个错误消息;其他时候,错误消息就是传递给error函数的值。

只要错误消息是一个字符串,Lua就会附加一些关于错误发生位置的信息(包含文件名比如stdin,及行号)。



local status,err = pcall(function () a = "a" + 1 end)
print(err)        --> stdin:1: attempt to perform arithmetic on a string value
local status,err = pcall(function() error("my error") end)
print(err)        -->stdin:1: my error



error函数还有第二个附加参数level,用于指出应由调用层级中的哪个(层)函数来报告当前的错误。

也就是说明了谁应该为此错误负责。比如一个函数,它的功能是检查传入的参数是否正确:



function foo (str)
    if type(str) ~= "string" then
        error("string expected")
    end
        <come code>
    end



然后有人这样传入错误参数地去调用:



foo({x=1})



由于foo是调用了error,所以Lua会认为是函数发生了错误。但实际上却是foo的调用者造成的错误。

为了纠正这个问题,就要告知error函数错误是发生在调用层级的第二层中(第一层是读函数)。



function foo(str)
    if type(str) ~= "string" then
        error("string expected",2)
    end
        <come code>
end



通常发生错误时,我们希望得到更多的调试信息。比如完整的调用栈,追溯到发生错误时的函数调用。

当pcall返回其错误信息时,它已经销毁了调用栈的部分内容(也就是从pcall到错误发生点的这部分调用)。

为此,我们必须在调用pcall之前保存好完整的栈信息。

xpcall很好地解决了这个问题。它接收第二个参数,一个消息处理函数。

一旦发生错误,Lua就会在调用栈展开前调用它,所以可以在这个函数中使用

debug lib去获取关于错误的额外信息了。

Lua的调试库提供了两个通用消息处理函数:

debug.debug:提供一个提示符,让用户检查错误的原因

debug.traceback:它会根据调用栈来构建一个扩展的错误消息。也可以在任何时候调用这个函数来获取当前执行的调用栈:



print (debug.trackback())