C++編程異常處理中try和throw以及catch語句的用法
若要在 C++ 中實現(xiàn)異常處理,你可以使用 try、throw 和 catch 表達(dá)式。
首先,使用 try 塊將可能引發(fā)異常的一個或多個語句封閉起來。
throw 表達(dá)式發(fā)出信號,異常條件(通常是錯誤)已在 try 塊中發(fā)生。你可以使用任何類型的對象作為 throw 表達(dá)式的操作數(shù)。該對象一般用于傳達(dá)有關(guān)錯誤的信息。大多數(shù)情況下,建議你使用 std::exception 類或標(biāo)準(zhǔn)庫中定義的派生類之一。如果其中的類不合適,建議你從 std::exception 派生自己的異常類。
若要處理可能引發(fā)的異常,請在 try 塊之后立即實現(xiàn)一個或多個 catch 塊。每個 catch 塊指定它能處理的異常類型。
以下示例將顯示 try 塊及其處理程序。假設(shè) GetNetworkResource() 通過網(wǎng)絡(luò)連接獲取數(shù)據(jù),并且兩個異常類型是從 std::exception 派生的用戶定義的類。請注意,異常由 catch 語句中的 const 引用捕獲。我們建議你通過值引發(fā)異常并通過常數(shù)引用將其捕獲。
MyData md; try { // Code that could throw an exception md = GetNetworkResource(); } catch (const networkIOException& e) { // Code that executes when an exception of type // networkIOException is thrown in the try block // ... // Log error message in the exception object cerr << e.what(); } catch (const myDataFormatException& e) { // Code that handles another exception type // ... cerr << e.what(); } // The following syntax shows a throw expression MyData GetNetworkResource() { // ... if (IOSuccess == false) throw networkIOException("Unable to connect"); // ... if (readError) throw myDataFormatException("Format error"); // ... }
備注
try 子句后的代碼是代碼的受保護(hù)部分。 throw 表達(dá)式將引發(fā)(即引起)異常。 catch 子句后的代碼塊是異常處理程序。如果 throw 和 catch 表達(dá)式中的類型兼容,該處理程序?qū)⒉东@引發(fā)的異常。有關(guān)管理 catch 塊中類型匹配的規(guī)則的列表,請參閱Catch 塊的計算方式 (C++)。如果 catch 語句指定省略號 (...) 而非類型,catch 塊將處理每種類型的異常。當(dāng)你使用 /EHa 選項編譯時,異??砂?C 結(jié)構(gòu)化異常和系統(tǒng)生成或應(yīng)用程序生成的異步異常,例如內(nèi)存保護(hù)、被零除和浮點沖突。由于 catch 塊按編程順序處理以查找匹配類型,所以盡量不要使用省略號處理程序來處理關(guān)聯(lián)的 try 塊。請謹(jǐn)慎使用 catch(...);除非 catch 塊知道如何處理捕獲的特定異常,否則禁止程序繼續(xù)執(zhí)行。 catch(...) 塊一般用于在程序停止執(zhí)行前記錄錯誤和執(zhí)行特殊的清理工作。
沒有操作數(shù)的 throw 表達(dá)式將重新引發(fā)當(dāng)前正在處理的異常。我們建議在重新引發(fā)異常時采用該形式,是因為這將保留原始異常的多態(tài)類型信息。此類表達(dá)式只應(yīng)在 catch 處理程序中或從 catch 處理程序調(diào)用的函數(shù)中使用。重新引發(fā)的異常對象是原始異常對象,而不是副本。
try { throw CSomeOtherException(); } catch(...) { // Catch all exceptions – dangerous!!! // Respond (perhaps only partially) to the exception, then // re-throw to pass the exception to some other handler // ... throw; }
Catch 塊的計算方式 (C++)
雖然通常建議您引發(fā)派生自 std::exception 的類型,但 C++ 使您能夠引發(fā)任何類型的異常??梢酝ㄟ^指定與引發(fā)的異常相同的類型的 catch 處理程序或通過可捕獲任何類型的異常的處理程序來捕獲 C++ 異常。
如果引發(fā)的異常的類型是類,它還具有基類(或類),則它可由接受異常類型的基類和對異常類型的基的引用的處理程序捕獲。請注意,當(dāng)異常由引用捕獲時,會將其綁定到實際引發(fā)的異常對象;否則,它將為一個副本(與函數(shù)的參數(shù)大致相同)。
引發(fā)異常時,將由以下類型的 catch 處理程序捕獲該異常:
- 可以接受任何類型的處理程序(使用省略號語法)。
- 接受與異常對象相同的類型的處理程序;由于它是副本,因此 const 和 volatile 修飾符將被忽略。
- 接受對與異常對象相同的類型的引用的處理程序。
- 接受對與異常對象相同的類型的 const 或 volatile 形式的引用的處理程序。
- 接受與異常對象相同的類型的基類的處理程序;由于它是副本,因此 const 和 volatile 修飾符將被忽略。基類的 catch 處理程序不得位于派生類的 catch 處理程序的前面。
- 接受對與異常對象相同的類型的基類的引用的處理程序。
- 接受與異常對象相同的類型的基類的 const 或 volatile 形式的引用的處理程序。
- 接受可通過標(biāo)準(zhǔn)指針轉(zhuǎn)換規(guī)則將引發(fā)的指針對象轉(zhuǎn)換為的指針的處理程序。
catch 處理程序出現(xiàn)的順序是有意義的,因為給定 try 塊的處理程序按它們的出現(xiàn)順序進(jìn)行檢查。例如,將基類的處理程序放置在派生類的處理程序的前面是錯誤的。 找到一個匹配的 catch 處理程序后,不會檢查后續(xù)處理程序。因此,省略號 catch 處理程序必須是其 try 塊的最后一個處理程序。例如:
// ... try { // ... } catch( ... ) { // Handle exception here. } // Error: the next two handlers are never examined. catch( const char * str ) { cout << "Caught exception: " << str << endl; } catch( CExcptClass E ) { // Handle CExcptClass exception here. }
在此示例中,省略號 catch 處理程序是已檢查的唯一處理程序。
相關(guān)文章
你只用do-while來實現(xiàn)循環(huán)?太浪費了
這篇文章主要介紹了你只用do-while來實現(xiàn)循環(huán)?太浪費了,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-12-12C++實現(xiàn)查找二叉樹中和為某一值的所有路徑的示例
這篇文章主要介紹了C++實現(xiàn)查找二叉樹中和為某一值的所有路徑的示例,文中的方法是根據(jù)數(shù)組生成二叉排序樹并進(jìn)行遍歷,需要的朋友可以參考下2016-02-02輕松實現(xiàn)C/C++各種常見進(jìn)制相互轉(zhuǎn)換
這篇文章主要介紹了輕松實現(xiàn)C/C++各種常見進(jìn)制相互轉(zhuǎn)換,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-11-11C語言數(shù)據(jù)結(jié)構(gòu)之串插入操作
這篇文章主要介紹了C語言數(shù)據(jù)結(jié)構(gòu)之串插入操作的相關(guān)資料,希望通過本文能幫助到大家,讓大家實現(xiàn)這樣的功能,需要的朋友可以參考下2017-10-10C++設(shè)計模式中控制反轉(zhuǎn)與依賴注入淺析
這篇文章主要介紹了C++設(shè)計模式中控制反轉(zhuǎn)與依賴注入,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2023-01-01