[erlang 001] erlang中的错误及异常处理
一、 erlang中的错误
1. 分类
1) 编译错误:主要是编译器检测出的代码语法错误;
2) 逻辑错误:是指程序没有完成预期的工作,属于开发人员的问题;
3) 运行时错误:是指erlang运行时抛出的错误,比如对非数据类型执行算术运算,erlang运行时会捕获异常,并抛出。在erlang中,这类异常的类型为error;
4) 用户代码生成的错误:是指通过exit/1或者throw/1生成。
2. 异常
我们把运行时错误以及用户抛出的错误称为异常(exception),他们具有三种类型:throw, error, exit。
1) error型异常,通过erlang:error/1, 2生成,用于抛出那些“崩溃错误”。这种异常应该是调用者不会真正意识到要去处理的那些致命错误;
2) throw型异常,通过throw/1生成,用于抛出一个调用者可能会捕获的异常。针对throw,必须为函数添加注释,说明他会抛出这个异常。调用者可以选择:忽略这些异常/对异常进行处理;
3) exit型异常,通过exit/1生成,当想要终止当前进程时,用这个函数。如果这个消息未被捕获,那么系统会向所有与当前进程连接的进程广播{'EXIT',Pid,Reason}消息
3. 异常的捕获
在erlang中,进程内的异常可以通过try, catch来进行捕获处理。
推荐使用try,其为新添加的语法。进程间的异常可以通过监督树(supervisor tree),监控进程(monitor)来实现。
如果是通过catch捕捉这三种异常,则返回的结果分别是:
- throw(Any) -> Term
- exit(Reason) -> {'EXIT',Reason}
- error(Reason) -> {'EXIT',{Reason,erlang:get_stacktrace()}}
4. 一些场景的报错类型
badarg:参数错误,参数格式或类型错误
badarith:算术表达式错误,算术表达式中含有错误的参数
{badmatch,V}:模式匹配错误,V指具体的发生匹配错误的数值
function_clause:函数子句错误,没有找到匹配的函数子句
{case_clause,V}:case匹配错误,没有找到匹配的case pattern
if_clause:if子句错误,没有找到为ture的if子句
{try_clause,V}:try匹配错误,执行try时,没有找到匹配的pattern
undef:函数未定义错误
{badfun,F}:函数错误
{badarity,F}:函数参数个数错误
timeout_value:超时参数错误,在receive.. after语法中,after对应的超时数据错误(应为不小于0的integer或infinity
noproc:Process 错误,Process不存在
{nocatch,V}:throw未被catch
system_limit:系统限制错误,某些性能或数据达到系统极限
二、try…catch语法
1. 语法
try FuncOrExpressionSeq of Pattern1 [when Guard1] -> Expressions1; Pattern2 [when Guard2] -> Expressions2; ... catch ExceptionType1: ExPattern1 [when ExGuard1] -> ExExpressions1; ExceptionType2: ExPattern2 [when ExGuard2] -> ExExpressions2; ... after AfterExpressions end
返回值:try表达式具有一个值;如果try FuncOrExpressionSeq后没有of部分,则默认为FuncOrExpressionSeq的返回值;
异常处理语句发生异常:如果在of部分或者catch部分,发生了异常,那么异常将不被处理,直接抛出;
catch语句块的ExceptionType1前内容省略的话,默认的错误标签是throw
after部分:after之后的代码是用来在FuncOrExpressionSeq结束后执行清理的,这段代码一定会被执行。并且FuncOrExpressionSeq的返回值会被丢弃。
2. 测试代码
代码部分:
-module(test_try). -compile([export_all]). -author('cheng litaocheng@gmail.com'). %% @spec test(F1, F2) -> Result %% @doc evaluate the F , use the try to catch all kinds of error %% F1 the Expression to be catch exception %% F2 the Expression evaluate in the catch section test(F1, F2) when (is_function(F1, 0) andalso is_function(F2, 0)) -> try F1() catch throw:X -> {{caught, throw, X}, F2()}; exit:X -> {{caught, exit, X}, F2()}; error:X -> {{caught, error, X}, F2()} after io:format("always evaluate the after body~n") end.
测试结果:
21> c(test_try). {ok,test_try} 22> test_try:test(fun() -> throw(hello) end, fun() -> ok end). always evaluate the after body {{caught,throw,hello},ok} 23> test_try:test(fun() -> exit(hello) end, fun() -> ok end). always evaluate the after body {{caught,exit,hello},ok} 24> test_try:test(fun() -> erlang:error(hello) end, fun() -> ok end). always evaluate the after body {{caught,error,hello},ok} 25> test_try:test(fun() -> erlang:error(hello) end, fun() -> throw(exception_in_catch) end). always evaluate the after body ** exception throw: exception_in_catch in function test_try:test/2
参考及整理自: