若要在 C++ 中實(shí)現(xiàn)異常處理,你可以使用 try、throw 和 catch 表達(dá)式。
首先,使用 try 塊將可能引發(fā)異常的一個(gè)或多個(gè)語(yǔ)句封閉起來(lái)。
throw 表達(dá)式發(fā)出信號(hào),異常條件(通常是錯(cuò)誤)已在 try 塊中發(fā)生。你可以使用任何類(lèi)型的對(duì)象作為 throw 表達(dá)式的操作數(shù)。該對(duì)象一般用于傳達(dá)有關(guān)錯(cuò)誤的信息。大多數(shù)情況下,建議你使用 std::exception 類(lèi)或標(biāo)準(zhǔn)庫(kù)中定義的派生類(lèi)之一。如果其中的類(lèi)不合適,建議你從 std::exception 派生自己的異常類(lèi)。
若要處理可能引發(fā)的異常,請(qǐng)?jiān)?try 塊之后立即實(shí)現(xiàn)一個(gè)或多個(gè) catch 塊。每個(gè) catch 塊指定它能處理的異常類(lèi)型。
以下示例將顯示 try 塊及其處理程序。假設(shè) GetNetworkResource() 通過(guò)網(wǎng)絡(luò)連接獲取數(shù)據(jù),并且兩個(gè)異常類(lèi)型是從 std::exception 派生的用戶定義的類(lèi)。請(qǐng)注意,異常由 catch 語(yǔ)句中的 const 引用捕獲。我們建議你通過(guò)值引發(fā)異常并通過(guò)常數(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 expressionMyData GetNetworkResource(){ // ... if (IOSuccess == false) throw networkIOException("Unable to connect"); // ... if (readError) throw myDataFormatException("Format error"); // ...} 備注
try 子句后的代碼是代碼的受保護(hù)部分。 throw 表達(dá)式將引發(fā)(即引起)異常。 catch 子句后的代碼塊是異常處理程序。如果 throw 和 catch 表達(dá)式中的類(lèi)型兼容,該處理程序?qū)⒉东@引發(fā)的異常。有關(guān)管理 catch 塊中類(lèi)型匹配的規(guī)則的列表,請(qǐng)參閱Catch 塊的計(jì)算方式 (C++)。如果 catch 語(yǔ)句指定省略號(hào) (...) 而非類(lèi)型,catch 塊將處理每種類(lèi)型的異常。當(dāng)你使用 /EHa 選項(xiàng)編譯時(shí),異??砂?C 結(jié)構(gòu)化異常和系統(tǒng)生成或應(yīng)用程序生成的異步異常,例如內(nèi)存保護(hù)、被零除和浮點(diǎn)沖突。由于 catch 塊按編程順序處理以查找匹配類(lèi)型,所以盡量不要使用省略號(hào)處理程序來(lái)處理關(guān)聯(lián)的 try 塊。請(qǐng)謹(jǐn)慎使用 catch(...);除非 catch 塊知道如何處理捕獲的特定異常,否則禁止程序繼續(xù)執(zhí)行。 catch(...) 塊一般用于在程序停止執(zhí)行前記錄錯(cuò)誤和執(zhí)行特殊的清理工作。
沒(méi)有操作數(shù)的 throw 表達(dá)式將重新引發(fā)當(dāng)前正在處理的異常。我們建議在重新引發(fā)異常時(shí)采用該形式,是因?yàn)檫@將保留原始異常的多態(tài)類(lèi)型信息。此類(lèi)表達(dá)式只應(yīng)在 catch 處理程序中或從 catch 處理程序調(diào)用的函數(shù)中使用。重新引發(fā)的異常對(duì)象是原始異常對(duì)象,而不是副本。
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 塊的計(jì)算方式 (C++)
雖然通常建議您引發(fā)派生自 std::exception 的類(lèi)型,但 C++ 使您能夠引發(fā)任何類(lèi)型的異常??梢酝ㄟ^(guò)指定與引發(fā)的異常相同的類(lèi)型的 catch 處理程序或通過(guò)可捕獲任何類(lèi)型的異常的處理程序來(lái)捕獲 C++ 異常。
如果引發(fā)的異常的類(lèi)型是類(lèi),它還具有基類(lèi)(或類(lèi)),則它可由接受異常類(lèi)型的基類(lèi)和對(duì)異常類(lèi)型的基的引用的處理程序捕獲。請(qǐng)注意,當(dāng)異常由引用捕獲時(shí),會(huì)將其綁定到實(shí)際引發(fā)的異常對(duì)象;否則,它將為一個(gè)副本(與函數(shù)的參數(shù)大致相同)。
引發(fā)異常時(shí),將由以下類(lèi)型的 catch 處理程序捕獲該異常:
- 可以接受任何類(lèi)型的處理程序(使用省略號(hào)語(yǔ)法)。
- 接受與異常對(duì)象相同的類(lèi)型的處理程序;由于它是副本,因此 const 和 volatile 修飾符將被忽略。
- 接受對(duì)與異常對(duì)象相同的類(lèi)型的引用的處理程序。
- 接受對(duì)與異常對(duì)象相同的類(lèi)型的 const 或 volatile 形式的引用的處理程序。
- 接受與異常對(duì)象相同的類(lèi)型的基類(lèi)的處理程序;由于它是副本,因此 const 和 volatile 修飾符將被忽略?;?lèi)的 catch 處理程序不得位于派生類(lèi)的 catch 處理程序的前面。
- 接受對(duì)與異常對(duì)象相同的類(lèi)型的基類(lèi)的引用的處理程序。
- 接受與異常對(duì)象相同的類(lèi)型的基類(lèi)的 const 或 volatile 形式的引用的處理程序。
- 接受可通過(guò)標(biāo)準(zhǔn)指針轉(zhuǎn)換規(guī)則將引發(fā)的指針對(duì)象轉(zhuǎn)換為的指針的處理程序。
catch 處理程序出現(xiàn)的順序是有意義的,因?yàn)榻o定 try 塊的處理程序按它們的出現(xiàn)順序進(jìn)行檢查。例如,將基類(lèi)的處理程序放置在派生類(lèi)的處理程序的前面是錯(cuò)誤的。 找到一個(gè)匹配的 catch 處理程序后,不會(huì)檢查后續(xù)處理程序。因此,省略號(hào) catch 處理程序必須是其 try 塊的最后一個(gè)處理程序。例如:
// ...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.} 在此示例中,省略號(hào) catch 處理程序是已檢查的唯一處理程序。



















