Symbian学习总结-基础篇-Symbian C++中的异常处理
本文章由杨芹勍原创,如需转摘请注明出处。谢谢!
在现代高级语言中,大多数语言都有对应处理异常的关键字,如:C++中的try/catch,Delphi中的Try/Except/Finally,这些关键字都被编译器直接的支持。
但是,当Symbian OS C++被发明的时候,C++中的关键字try/catch/finally还没有被标准支持。而且照Symbian官方的话来说,C++的异常处理机制太笨重,不够轻量级,他们设计了一套效率更加高,更适合于小型的、内存受限的设备的异常处理机制。
一、 回顾历史
在标准C++中,我们使用如下方法来处理异常:
1 2 3 4 5 6 7 8 9 |
void DoTest() { try { int n = 1 / 0; } catch (...) { // do something... throw; } } |
ps:标准C++并不支持finally。
很明显,程序执行到第2行的时候回抛出异常,然后跳转到第5行去执行异常处理,在第6行使用throw将异常抛给外部。
二、 转变观念
Symbian OS C++中的异常处理和标准C++的异常处理大同小异。首先我们先将标准C++中的异常处理代码使用以下方式编写
1 2 3 4 5 6 7 8 9 10 11 12 13 |
void DoDiv() { int n = 1 / 0; } void DoTest() { try { DoDiv(); } catch(...) { // do Something... throw; } } |
首先,把可能会出错的语句提取方法到DoDiv中,在DoTest中调用DoDiv时放在异常处理捕获语句中调用。
三、 Symbian中的异常处理
在此,我们按照Symbian C++的写法翻译以上错误处理代码:
1 2 3 4 5 6 7 8 9 10 |
void DoDivL() { int n = 1 / 0; } void DoTest() { TRAPD(error, DoDivL()); if (error) { User::Leave(); } } |
第一行的函数DoDivL,后缀“L”的意思是:此函数可能会抛出异常。这只是Symbian C++的一种命名规范,让程序员一看到L就会想到要捕获异常。不写后缀“L”对程序执行不会有影响。
第6行的TRAPD是Symbian C++的错误处理截获宏,调用原型为TRAPD(_r, _s)。在TRAPD中的第2个参数_s为可能会抛出异常的语句,第一个参数_r为错误返回代码,如果调用_s出现错误,则将错误代码存放入_r中,程序继续执行第7行。
User::Leave()相当于c++中的throw,将异常往外抛出。
Symbian C++提供了很多User::Leave**方法,如:User::LeaveIfError()、User::LeaveNoMemory()、User::LeaveIfNull(),其中,User::LeaveIfError()相当于以上代码的第7至第9行。
四、宏TRAP和TRAPD的区别:
“D”表示Declare的意思,宏TRAP的调用原型为TRAP(_r, _s)。从以上代码第7行可以看出,变量error已经被TRAPD定义,可以直接使用。而TRAP宏并不会定义_r参数变量,使用TRAP宏,DoTest函数将使用以下方式编写:
1 2 3 4 5 6 7 |
void DoTest() { TInt error; TRAP(error, DoDivL()); if (error) { User::Leave(); } } |
对于默认栈只有8K大小的程序来说,我们应该优先使用TRAP宏。