LUA的异常处理

LUA的异常处理
1、可以通过调用error函数显式地抛出错误,error的参数是要抛出的错误信息。

print "enter a number:"

n = io.read("*number")

if not n then

error("invalid input")

end

2、可以通过assert函数显示错误

print "enter a number:"

n = assert(io.read("*number"), "invalid input")

assert首先检查第一个参数,若没问题,assert不做任何事情;否则,assert以第二个参数作为错误信息抛出。第二个参数是可选的。注意,assert会首先处理两个参数,然后才调用函数,所以下面代码,无论n是否为数字,字符串连接操作总会执行:

n = io.read()

assert(tonumber(n), "invalid input: " .. n .. " is not a number")

当函数遇到异常有两个基本的动作:返回错误代码或者抛出错误。选择哪一种方式,没有固定的规则,不过基本的原则是:对于程序逻辑上能够避免的异常,以抛出错误的方式处理之,否则返回错误代码。

3、有时候不需要在Lua进行错误处理,一般有应用来完成。通常应用要求Lua运行一段chunk,如果发生异常,应用根据Lua返回的错误代码进行处理。在控制台模式下的Lua解释器如果遇到异常,打印出错误然后继续显示提示符等待下一个命令。

如果在Lua中需要处理错误,需要使用pcall函数封装你的代码。

假定你想运行一段Lua代码,这段代码运行过程中可以捕捉所有的异常和错误。

第一步:将这段代码封装在一个函数内

function foo ()

...

if unexpected_condition then error() end

...

print(a[i]) -- potential error: `a' may not be a table

...

end

第二步:使用pcall调用这个函数

if pcall(foo) then

-- no errors while running `foo'

...

else

-- `foo' raised an error: take appropriate actions

...

end

当然也可以用匿名函数的方式调用pcall

if pcall(function () ... end) then ...

else ...

pcall在保护模式(protected mode)下执行函数内容,同时捕获所有的异常和错误。若一切正常,pcall返回true以及“被执行函数”的返回值;否则返回nil和错误信息。

错误信息不一定仅为字符串(下面的例子是一个table),传递给error的任何信息都会被pcall返回:

local status, err = pcall(function () error({code=121}) end)

print(err.code) --> 121

这种机制提供了强大的能力,足以应付Lua中的各种异常和错误情况。我们通过error抛出异常,然后通过pcall捕获之。

4、错误跟踪

虽然你可以使用任何类型的值作为错误信息,通常情况下,我们使用字符串来描述遇到的错误。如果遇到内部错误(比如对一个非table的值使用索引下标访问)Lua将自己产生错误信息,否则Lua使用传递给error函数的参数作为错误信息。不管在什么情况下,Lua都尽可能清楚的描述问题发生的缘由。

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

例子中错误信息给出了文件名(stdin)与行号。

函数error还可以有第二个参数,表示错误发生的层级。比如,你写了一个函数用来检查“error是否被正确调用”:

function foo (str)

if type(str) ~= "string" then

error("string expected")

end

...

end

可有人这样调用此函数:

foo({x=1})

Lua会指出发生错误的是foo而不是error,实际上,错误是调用error时产生的。为了纠正这个问题,修改前面的代码让error报告错误发生在第二级(你自己的函数是第一级)如下:

function foo (str)

if type(str) ~= "string" then

error("string expected", 2)

end

...

end

当错误发生的时候,我们常常希望了解详细的信息,而不仅是错误发生的位置。若能了解到“错误发生时的栈信息”就好了,但pcall返回错误信息时,已经释放了保存错误发生情况的栈信息。因此,若想得到tracebacks,我们必须在pcall返回以前获取。Lua提供了xpcall来实现这个功能,xpcall接受两个参数:调用函数、错误处理函数。当错误发生时,Lua会在栈释放以前调用错误处理函数,因此可以使用debug库收集错误相关信息。有两个常用的debug处理函数:debug.debugdebug.traceback,前者给出Lua的提示符,你可以自己动手察看错误发生时的情况;后者通过traceback创建更多的错误信息,也是控制台解释器用来构建错误信息的函数。你可以在任何时候调用debug.traceback获取当前运行的traceback信息:

print(debug.traceback())

推荐阅读