[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

 

参考及整理自:

[1] http://erlangdisplay.iteye.com/blog/315417 

[2] http://blog.sina.com.cn/s/blog_53a5047b01018wp5.html

posted @ 2015-04-08 16:09  大师兄啊哈  阅读(1820)  评论(0编辑  收藏  举报