本文章由楊芹勍原創,如需轉摘請註明出處。謝謝!
在現代進階語言中,大多數語言都有對應處理異常的關鍵字,如:C++中的try/catch,Delphi中的Try/Except/Finally,這些關鍵字都被編譯器直接的支援。
但是,當Symbian OS C++被發明的時候,C++中的關鍵字try/catch/finally還沒有被標準支援。而且照Symbian官方的話來說,C++的異常處理機制太笨重,不夠輕量級,他們設計了一套效率更加高,更適合於小型的、記憶體受限的裝置的異常處理機制。
一、 回顧曆史
在標準C++中,我們使用如下方法來處理異常:
123456789 |
void DoTest() {try {int n = 1 / 0;}catch (...) {// do something...throw;}} |
ps:標準C++並不支援finally。
很明顯,程式執行到第2行的時候回拋出異常,然後跳轉到第5行去執行異常處理,在第6行使用throw將異常拋給外部。
二、 轉變觀念
Symbian OS C++中的異常處理和標準C++的異常處理大同小異。首先我們先將標準C++中的異常處理代碼使用以下方式編寫
12345678910111213 |
void DoDiv() {int n = 1 / 0;}void DoTest() {try {DoDiv();}catch(...) {// do Something...throw;}} |
首先,把可能會出錯的語句提取方法到DoDiv中,在DoTest中調用DoDiv時放在異常處理捕獲語句中調用。
三、 Symbian中的異常處理
在此,我們按照Symbian C++的寫法翻譯以上錯誤處理代碼:
12345678910 |
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函數將使用以下方式編寫:
1234567 |
void DoTest() {TInt error;TRAP(error, DoDivL());if (error) {User::Leave();}} |
對於預設棧只有8K大小的程式來說,我們應該優先使用TRAP宏。