C++異常處理的方式總結(jié)
個(gè)詞,今天就來做詳細(xì)的介紹,文中通過代碼示例給大家介紹的非常詳細(xì),具有一定參考價(jià)值,需要的朋友可以參考下
1. 前言
C++有一套獨(dú)立的異常處理機(jī)制,相信大家一定聽說過try,catch這兩個(gè)詞,今天就來做詳細(xì)的介紹
本篇文章著重講解C++異常處理的方式,三個(gè)關(guān)鍵字,tyr,catch,throw,并且介紹異常的用法和自定義體系的異常以及智能指針在異常處理中的使用場景.其中,會復(fù)習(xí)C語言異常處理的方式
2. C語言處理異常的方式
最經(jīng)典的處理方式:使用assert
assert的缺陷:
如果在代碼中使用assert,則只在debug模式下有效,在release模式下會失效.并且只要有錯(cuò)誤就會直接終止程序,這明顯不符合實(shí)際,比如說在使用微信時(shí),由于網(wǎng)絡(luò)問題信息沒發(fā)出去,這時(shí)直接將微信程序終止了,這樣做會被亂棍打死!
C語言還能用錯(cuò)誤碼返回異常信息
錯(cuò)誤碼errno的缺陷:
返回的錯(cuò)誤碼是一個(gè)數(shù)字,程序員還需去查表來得知這個(gè)錯(cuò)誤碼是什么意思,并且就算查找了錯(cuò)誤碼的信息,可能它說的不清楚,也不好看錯(cuò)誤信息
綜上所述,C語言處理異常的方式還是不夠完美,于是祖師爺寫了一套自己的
3. C++異常概念
當(dāng)一個(gè)函數(shù)發(fā)現(xiàn)自己無法處理的錯(cuò)誤時(shí)就可以拋出異常,讓函數(shù)的直接或間接的調(diào)用者處理這個(gè)錯(cuò)誤。
throw:
當(dāng)問題出現(xiàn)時(shí),程序會拋出一個(gè)異常。這是通過使用 throw 關(guān)鍵字來完成的catch:
在您想要處理問題的地方,通過異常處理程序捕獲異常.catch 關(guān)鍵字用于捕獲異常try:
try 塊中的代碼標(biāo)識將被激活的特定異常,它后面通常跟著一個(gè)或多個(gè) catch 塊。
使用方法:
try{ int a,b; cin>>a>>b; if(b == 0) throw "除0錯(cuò)誤" cout<<(a/b)<<endl; } catch(string str) { //...... }
如果有一個(gè)塊拋出一個(gè)異常,捕獲異常的方法會使用 try 和 catch 關(guān)鍵字。try 塊中放置可能拋出異常的代碼,try 塊中的代碼被稱為保護(hù)代碼,一旦throw后,會直接跳到catch的位置,后面的代碼不會執(zhí)行
4. 異常的拋出和匹配原則
異常的拋出和匹配有以下機(jī)制:
拋出的內(nèi)容和捕捉的內(nèi)容要一致
如果你在throw時(shí)拋出一個(gè)字符串,但是在catch捕獲異常時(shí)的參數(shù)卻寫的是整數(shù)那么這個(gè)拋出的異常就不會去這個(gè)catch
void a(){throw "測試中";} void b(){a()} void c(){b()} int main() { try{ c(); } catch(string str) {}
try可以嵌套多層
try和catch不僅僅可以在一個(gè)作用域使用,還可以在最外層try,然后嵌套多層函數(shù),在最里面的函數(shù)throw!
void a(){throw "測試中";} void b(){a()} void c(){b()} int main() { try{ c(); } catch(string str) {}
throw和catch遵循就近原則
若寫了多個(gè)catch,并且這些catch都和throw的內(nèi)容匹配,則會跳轉(zhuǎn)到與throw最近的內(nèi)個(gè)catch中!
double Division(int a, int b) { // 當(dāng)b == 0時(shí)拋出異常 if (b == 0) throw "除0錯(cuò)誤!"; else return ((double)a / (double)b); } void Func() { int len, time; cin >> len >> time; cout << Division(len, time) << endl; catch (const char* errmsg) {//這個(gè)catch與throw相距最近,會優(yōu)先到這兒 cout << errmsg << "111" << endl; } } int main() { try { Func(); } catch (const char* errmsg) { cout << errmsg << "222" << endl; } return 0; }
catch(...)可以捕獲任意類型的異常
在公司寫大工程的時(shí)候,會和很多同事合作寫代碼,大家都會拋出異常,但是你不能確定是不是所有人拋出的類型你都有相應(yīng)的catch可以接收,若拋出一個(gè)異常沒有被捕獲會直接報(bào)錯(cuò),所以…的作用很明顯,用來兜底!一般用于接收一些未知異常
double Division(int a, int b) { // 當(dāng)b == 0時(shí)拋出異常 if (b == 0) throw "除0錯(cuò)誤!"; else return ((double)a / (double)b); } void Func() { int len, time; cin >> len >> time; cout << Division(len, time) << endl; catch (const char* errmsg) {//這個(gè)catch與throw相距最近,會優(yōu)先到這兒 cout << errmsg << "111" << endl; } } int main() { try { Func(); } catch (const char* errmsg) { cout << errmsg << "222" << endl; } catch(...){ cout<<"未知異常"<<endl; } return 0; }
基類可以接受拋出的子類對象
拋出和捕獲有一個(gè)例外,那就是可以拋出子類對象,用基類捕獲,這個(gè)在實(shí)際場景中非常實(shí)用,我們會在后面詳談
5. 異常的重新拋出
有可能在捕獲異常時(shí),一次捕獲不能完全解決問題,比如我們想在main函數(shù)中處理所有的異常,在非main函數(shù)中打印一下異常信息然后再將異常拋到main中統(tǒng)一做處理
double Division(int a, int b) { // 當(dāng)b == 0時(shí)拋出異常 if (b == 0) { throw "Division by zero condition!"; } return (double)a / (double)b; } void Func() { // 這里可以看到如果發(fā)生除0錯(cuò)誤拋出異常,另外下面的array沒有得到釋放。 // 所以這里捕獲異常后并不處理異常,異常還是交給外面處理,這里捕獲了再 // 重新拋出去。 int* array = new int[10]; try { int len, time; cin >> len >> time; cout << Division(len, time) << endl; } catch (...) { cout << "delete []" << array << endl; delete[] array; throw; } cout << "delete []" << array << endl; delete[] array; } int main() { try { Func(); } catch (const char* errmsg) { cout << errmsg << endl; } return 0; }
在上面的場景中,如果拋出異常,就會直接走到catch處,不會調(diào)用delete釋放釋放在堆上開辟的空間,就會有問題,所以第一次catch時(shí)先處理釋放空間的問題然后將異常再次拋出后再處理異常問題
6. RAII思想在異常體系中的使用
在異常體系中在堆上申請空間,或者打開某個(gè)問題時(shí)常容易出問題,因?yàn)槎焉祥_辟的空間要顯示調(diào)用delete處理而打開的文件也要顯示調(diào)用fclose關(guān)閉所以一旦發(fā)生異常就會直接跳轉(zhuǎn)到catch的位置,有可能直接忽略了釋放函數(shù)這也就是導(dǎo)致了資源并沒有被釋放!
在異常體系中最好使用RAII思想申請資源
即使拋出異常后直接跳到catch也沒問題
當(dāng)出了對象作用域會自動(dòng)調(diào)用析構(gòu)釋放!
double Division(int a, int b) { // 當(dāng)b == 0時(shí)拋出異常 if (b == 0) { throw "Division by zero condition!"; } return (double)a / (double)b; } void Func() { shared_ptr<int> array(new int(10)); try { int len, time; cin >> len >> time; cout << Division(len, time) << endl; } } int main() { try { Func(); } catch (const char* errmsg) { cout << errmsg << endl; } return 0; }
7. 自定義異常體系
實(shí)際使用中很多公司都會自定義自己的異常體系進(jìn)行規(guī)范的異常管理,因?yàn)橐粋€(gè)項(xiàng)目中如果大家隨意拋異常,那么外層的調(diào)用者基本就沒辦法玩了,所以實(shí)際中都會定義一套繼承的規(guī)范體系。這樣大家拋出的都是繼承的派生類對象,捕獲一個(gè)基類就可以了
不同的部分可以拋出不同的異常,然后在總的main函數(shù)中使用基類捕獲所有的異常再來進(jìn)行特殊的處理
8. C++標(biāo)準(zhǔn)庫的異常體系
這里的內(nèi)容屬于了解范疇,用幾張圖帶大家了解一下:
實(shí)際中都是我們自己去實(shí)現(xiàn)一個(gè)異常體系
因?yàn)镃++庫做的并不好
9. 總結(jié)以及拓展
異??傮w而言,利大于弊,所以工程中我們還是鼓勵(lì)使用異常的。另外OO的語言基本都是用異常處理錯(cuò)誤,這也可以看出這是大勢所趨。
除此之外,異常還有一套規(guī)范,因?yàn)楸容^雞肋,所以放在了最后來介紹:
- 異常規(guī)格說明的目的是為了讓函數(shù)使用者知道該函數(shù)可能拋出的異常有哪些。 可以在函數(shù)的后面接throw(類型),列出這個(gè)函數(shù)可能拋擲的所有異常類型。
- 函數(shù)的后面接throw(),表示函數(shù)不拋異常。
- 若無異常接口聲明,則此函數(shù)可以拋擲任何類型的異常。
// 這里表示這個(gè)函數(shù)會拋出A/B/C/D中的某種類型的異常 void fun() throw(A,B,C,D); // 這里表示這個(gè)函數(shù)只會拋出bad_alloc的異常 void* operator new (std::size_t size) throw (std::bad_alloc); // 這里表示這個(gè)函數(shù)不會拋出異常 void* operator delete (std::size_t size, void* ptr) throw(); // C++11 中新增的noexcept,表示不會拋異常 thread() noexcept; thread (thread&& x) noexcept;
之所以比較雞肋是因?yàn)榫退隳銓懥薾oexcept,再拋出異常,在某些編譯器也不會報(bào)錯(cuò)
以上就是C++異常處理的方式總結(jié)的詳細(xì)內(nèi)容,更多關(guān)于C++異常處理方式的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Visual Studio Code 從簡介、安裝到配置所需插件詳細(xì)介紹
這篇文章給大家介紹到vs與vs code的區(qū)別,并且會詳細(xì)介紹vscode的安裝步驟,和我所了解過的插件配置,感興趣的朋友跟隨小編一起看看吧2020-03-03C++實(shí)現(xiàn)統(tǒng)計(jì)代碼運(yùn)行時(shí)間計(jì)時(shí)器的簡單實(shí)例
這篇文章主要介紹了 C++實(shí)現(xiàn)統(tǒng)計(jì)代碼運(yùn)行時(shí)間計(jì)時(shí)器的簡單實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-07-07C++通過msxml調(diào)用webservice示例分享
這篇文章主要介紹了C++通過msxml調(diào)用webservice示例分享,需要的朋友可以參考下2014-03-03C++實(shí)現(xiàn)LeetCode(66.加一運(yùn)算)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(66.加一運(yùn)算),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07數(shù)據(jù)結(jié)構(gòu) 中數(shù)制轉(zhuǎn)換(棧的應(yīng)用)
這篇文章主要介紹了數(shù)據(jù)結(jié)構(gòu) 中數(shù)制轉(zhuǎn)換(棧的應(yīng)用)的相關(guān)資料,需要的朋友可以參考下2017-06-06