異常是程序在執(zhí)行期間產(chǎn)生的問題况木。
C++ 異常是指在程序運行時發(fā)生的特殊情況垒拢,比如嘗試除以零的操作。
異常提供了一種轉(zhuǎn)移程序控制權(quán)的方式火惊。C++ 異常處理涉及到三個關(guān)鍵字:try求类、catch、throw屹耐。
- throw: 當問題出現(xiàn)時尸疆,程序會拋出一個異常。這是通過使用 throw 關(guān)鍵字來完成的。
- catch: 在您想要處理問題的地方寿弱,通過異常處理程序捕獲異常犯眠。catch 關(guān)鍵字用于捕獲異常。
- try: try塊中的代碼標識將被激活的特定異常症革。它后面通常跟著一個或多個 catch 塊筐咧。
如果有一個塊拋出一個異常,捕獲異常的方法會使用 try 和 catch 關(guān)鍵字噪矛。try 塊中放置可能拋出異常的代碼量蕊,try 塊中的代碼被稱為保護代碼。使用 try/catch 語句的語法如下所示:
try
{
// 保護代碼
}catch( ExceptionName e1 )
{
// catch 塊
}catch( ExceptionName e2 )
{
// catch 塊
}catch( ExceptionName eN )
{
// catch 塊
}
如果 try 塊在不同的情境下會拋出不同的異常艇挨,這個時候可以嘗試羅列多個 catch 語句危融,用于捕獲不同類型的異常。
拋出異常
您可以使用 throw 語句在代碼塊中的任何地方拋出異常雷袋。throw 語句的操作數(shù)可以是任意的表達式吉殃,表達式的結(jié)果的類型決定了拋出的異常的類型。
以下是嘗試除以零時拋出異常的實例:
double division(int a, int b)
{
if( b == 0 )
{
throw "Division by zero condition!";
}
return (a/b);
}
捕獲異常
catch 塊跟在 try 塊后面楷怒,用于捕獲異常蛋勺。您可以指定想要捕捉的異常類型,這是由 catch 關(guān)鍵字后的括號內(nèi)的異常聲明決定的鸠删。
try
{
// 保護代碼
}catch( ExceptionName e )
{
// 處理 ExceptionName 異常的代碼
}
上面的代碼會捕獲一個類型為 ExceptionName 的異常抱完。如果您想讓 catch 塊能夠處理 try 塊拋出的任何類型的異常,則必須在異常聲明的括號內(nèi)使用省略號 ...刃泡,如下所示:
try
{
// 保護代碼
}catch(...)
{
// 能處理任何異常的代碼
}
下面是一個實例巧娱,拋出一個除以零的異常,并在 catch 塊中捕獲該異常烘贴。
實例
#include <iostream>
using namespace std;
double division(int a, int b)
{
if( b == 0 )
{
throw "Division by zero condition!";
}
return (a/b);
}
int main ()
{
int x = 50;
int y = 0;
double z = 0;
try {
z = division(x, y);
cout << z << endl;
}catch (const char* msg) {
cerr << msg << endl;
}
return 0;
}
由于我們拋出了一個類型為 const char* 的異常禁添,因此,當捕獲該異常時桨踪,我們必須在 catch 塊中使用 const char*老翘。當上面的代碼被編譯和執(zhí)行時,它會產(chǎn)生下列結(jié)果:
Division by zero condition!
C++ 標準的異常
C++ 提供了一系列標準的異常锻离,定義在 <exception> 中铺峭,我們可以在程序中使用這些標準的異常。它們是以父子類層次結(jié)構(gòu)組織起來的汽纠,如下所示:
下表是對上面層次結(jié)構(gòu)中出現(xiàn)的每個異常的說明:
異常 | 描述
- |
std::exception | 該異常是所有標準 C++ 異常的父類卫键。
std::bad_alloc | 該異常可以通過 new 拋出虱朵。
std::bad_cast | 該異忱蚵可以通過 dynamic_cast 拋出啤呼。
std::bad_exception | 這在處理 C++ 程序中無法預期的異常時非常有用。
std::bad_typeid | 該異衬馗ぃ可以通過 typeid 拋出官扣。
std::logic_error | 理論上可以通過讀取代碼來檢測到的異常。
std::domain_error | 當使用了一個無效的數(shù)學域時羞福,會拋出該異常惕蹄。
std::invalid_argument | 當使用了無效的參數(shù)時,會拋出該異常治专。
std::length_error | 當創(chuàng)建了太長的 std::string 時卖陵,會拋出該異常。
std::out_of_range | 該異痴欧澹可以通過方法拋出泪蔫,例如 std::vector 和 std::bitset<>::operator。
std::runtime_error | 理論上不可以通過讀取代碼來檢測到的異常喘批。
std::overflow_error | 當發(fā)生數(shù)學上溢時撩荣,會拋出該異常。
std::range_error | 當嘗試存儲超出范圍的值時饶深,會拋出該異常餐曹。
std::underflow_error | 當發(fā)生數(shù)學下溢時,會拋出該異常敌厘。
定義新的異常
您可以通過繼承和重載 exception 類來定義新的異常台猴。下面的實例演示了如何使用 std::exception 類來實現(xiàn)自己的異常:
實例
#include <iostream>
#include <exception>
using namespace std;
struct MyException : public exception
{
const char * what () const throw ()
{
return "C++ Exception";
}
};
int main()
{
try
{
throw MyException();
}
catch(MyException& e)
{
std::cout << "MyException caught" << std::endl;
std::cout << e.what() << std::endl;
}
catch(std::exception& e)
{
//其他的錯誤
}
}
這將產(chǎn)生以下結(jié)果:
MyException caught
C++ Exception
在這里,what() 是異常類提供的一個公共方法俱两,它已被所有子異常類重載饱狂。這將返回異常產(chǎn)生的原因。
代碼鏈接:https://github.com/karst87/cpp/tree/master/learning/com.runoob