




版權說明:本文檔由用戶提供并上傳,收益歸屬內容提供方,若內容存在侵權,請進行舉報或認領
文檔簡介
第七講異常處理異常處理異常處理標準程序庫異常處理綜合實例異常安全性原則智能指針異常處理的基本思想異常處理函數f()捕獲并處理異常函數h()引發異常函數g()……調用者異常傳播方向調用關系發現錯誤的函數往往不具備處理錯誤的能力,這時它就引發一個異常,希望調用者捕獲并處理這個錯誤。如果調用者也不能處理這個錯誤,還可以繼續傳遞給上級調用者去處理,這種傳播會一直繼續到異常被處理為止。如果程序始終沒有處理這個異常,最終它會被傳到C++運行系統,運行系統捕獲異常后通常只是簡單地終止這個程序。異常處理的語法異常處理拋擲異常的程序段......throw表達式;......捕獲并處理異常的程序段try
復合語句catch(異常聲明)復合語句catch(異常聲明)復合語句
…保護段異常處理程序C++的異常處理機制使得異常的引發和處理不必在同一函數中,這樣底層的函數可以集中精力解決具體問題,而不必過多地考慮對異常的處理異常是一種包含錯誤信息的對象需要拋出的異常對象或變量即拋出異常的代碼區域包含異常類型和異常參數,與函數形參類似如果異常聲明是(…),表示該子句可以處理所有類型的異常,這樣的子句必須放在最后。異常處理的語法異常處理若有異常則(系統或用戶)通過throw操作創建一個異常對象并拋擲將可能拋出異常的程序段嵌在try塊之中??刂仆ㄟ^正常的順序執行到達try語句,然后執行try塊內的保護段如果在保護段執行期間沒有引起異常,那么跟在try塊后的catch子句就不執行。程序從try塊后跟隨的最后一個catch子句后面的語句繼續執行下去否則,catch子句按其在try塊后出現的順序被檢查。匹配的catch子句將捕獲并處理異常(或繼續拋擲異常)如果當前函數沒有匹配的catch子句,或異常拋出點本身不在任何try子句內,則結束當前函數的執行,回到當前函數的調用點,把調用點作為異常的拋出點,然后重復這一過程,直到異常成功地被一個catch語句捕獲如果匹配的處理器始終未找到,則運行庫函數terminate將被自動調用,其缺省功能是調用abort終止程序處理除零異常異常處理#include<iostream>usingnamespacestd;intdivide(intx,inty){ if(y==0)
throwx; returnx/y;}intmain(){ try { cout<<"5/2="<<divide(5,2)<<endl; cout<<"8/0="<<divide(8,0)<<endl; cout<<"7/1="<<divide(7,1)<<endl; }catch(inte) { cout<<e<<"isdividedbyzero!"<<endl; } cout<<"Thatisok."<<endl; return0;}結果如下:5/2=28isdividedbyzero!Thatisok.異常接口聲明異常處理可以在函數的聲明中列出這個函數可能拋擲的所有異常類型例如:
voidfun()throw(A,B,C,D);若無異常接口聲明,則此函數可以拋擲任何類型的異常不拋擲任何類型異常的函數聲明如下:voidfun()throw();表明函數fun只能拋擲類型A、B、C、D及其子類型的異常異常處理中的構造與析構異常處理找到一個匹配的catch異常處理后初始化異常參數將從對應的try塊開始到異常被拋擲處之間構造(且尚未析構)的所有自動對象(即位于堆棧中的對象)進行析構。這一過程稱為棧的解旋(unwinding)從最后一個catch子句之后開始恢復執行示例:使用帶析構語義的類的C++異常處理異常處理#include<iostream>#include<string>usingnamespacestd;classMyException{public: MyException(conststring&message):message(message){} ~MyException(){}
conststring&getMessage()const{returnmessage;}private: stringmessage;};classDemo{public: Demo(){cout<<"ConstructorofDemo"<<endl;} ~Demo(){cout<<"DestructorofDemo"<<endl;}};示例:使用帶析構語義的類的C++異常處理異常處理voidfunc()throw(MyException){ Demod; cout<<"ThrowMyExceptioninfunc()"<<endl;
throwMyException("exceptionthrownbyfunc()");}intmain(){ cout<<"Inmainfunction"<<endl; try { func(); }catch(MyException&e) { cout<<"Caughtanexception:"<<e.getMessage()<<endl; } cout<<"Resumetheexecutionofmain()"<<endl; return0;}示例:使用帶析構語義的類的C++異常處理異常處理結果如下:InmainfunctionConstructorofDemoThrowMyExceptioninfunc()DestructorofDemoCaughtanexception:exceptionthrownbyfunc()Resumetheexecutionofmain()異常處理異常處理標準程序庫異常處理綜合實例異常安全性原則智能指針標準程序庫異常處理range_erroroverflow_errorunderflow_errorexceptiondomain_errorinvalid_argumentlength_errorios_base::failurelogic_errorbad_typeidbad_exceptionbad_castruntime_errorbad_allocout_of_range標準程序庫的異常類標準程序庫異常處理exception:標準程序庫異常類的公共基類logic_error表示可以在程序中被預先檢測到的異常如果小心地編寫程序,這類異常能夠避免runtime_error表示難以被預先檢測的異常三角形面積計算標準程序庫異常處理編寫一個計算三角形面積的函數,函數的參數為三角形三邊邊長a、b、c,可以用Heron公式計算:設,則三角形面積
#include<iostream>#include<cmath>#include<stdexcept>usingnamespacestd;//給出三角形三邊長,計算三角形面積doublearea(doublea,doubleb,doublec)throw(invalid_argument){ //判斷三角形邊長是否為正
if(a<=0||b<=0||c<=0)
throwinvalid_argument("thesidelengthshouldbepositive"); //判斷三邊長是否滿足三角不等式
if(a+b<=c||b+c<=a||c+a<=b)
throwinvalid_argument("thesidelengthshouldfitthetriangleinequation"); //由Heron公式計算三角形面積
doubles=(a+b+c)/2; returnsqrt(s*(s-a)*(s-b)*(s-c));}三角形面積計算標準程序庫異常處理intmain(){ doublea,b,c; //三角形三邊長
cout<<"Pleaseinputthesidelengthsofatriangle:"; cin>>a>>b>>c; try { doubles=area(a,b,c); //嘗試計算三角形面積
cout<<"Area:"<<s<<endl; }catch(exception&e) { cout<<"Error:"<<e.what()<<endl; } return0;}三角形面積計算標準程序庫異常處理運行結果1:Pleaseinputthesidelengthsofatriangle:345Area:6運行結果2:Pleaseinputthesidelengthsofatriangle:055Error:thesidelengthshouldbepositive運行結果2:Pleaseinputthesidelengthsofatriangle:124Error:thesidelengthshouldfitthetriangleinequation三角形面積計算標準程序庫異常處理異常處理異常處理標準程序庫異常處理綜合實例異常安全性原則智能指針綜合實例——對個人銀行賬戶管理程序的改進本例中,在構造或輸入一個Date對象時如發生了錯誤,直接使用標準程序庫中的runtime_error構造異常并拋出;在賬戶類中如發生了錯誤,希望異常信息能夠標識是哪個賬戶發生了錯誤。本程序中創建了一個類AccountException,該類從runtime_error派生,該類中保存了一個Account型常指針,指向發生錯誤的賬戶,這樣在主函數中,輸出錯誤信息的同時也可以將賬號輸出。綜合實例——對個人銀行賬戶管理程序的改進//date.cpp,僅列出與以前不同的內容,下同#include"date.h"#include<iostream>#include<stdexcept>usingnamespacestd;Date::Date(intyear,intmonth,intday):year(year),month(month),day(day){ if(day<=0||day>getMaxDay())
throwruntime_error("Invaliddate"); intyears=year-1; totalDays=years*365+years/4-years/100+years/400+DAYS_BEFORE_MONTH[month-1]+day; if(isLeapYear()&&month>2)totalDays++;}istream&operator>>(istream&in,Date&date){ intyear,month,day; charc1,c2; in>>year>>c1>>month>>c2>>day; if(c1!='-'||c2!='-')
throwruntime_error("Badtimeformat");
date=Date(year,month,day); returnin;}綜合實例——對個人銀行賬戶管理程序的改進//account.h#ifndef__ACCOUNT_H__#define__ACCOUNT_H__#include"date.h"#include"accumulator.h"#include<string>#include<map>#include<istream>#include<stdexcept>//account.h中增加了以下類,其它各類的定義與之前相同,不再重復給出classAccountException:publicstd::runtime_error{private: constAccount*account;public: AccountException(constAccount*account,conststd::string&msg) :runtime_error(msg),account(account){}
constAccount*getAccount()const{returnaccount;}};#endif//__ACCOUNT_H__綜合實例——對個人銀行賬戶管理程序的改進//account.cpp中僅以下成員函數的實現與之前不同,其它內容皆與之相同voidAccount::error(conststring&msg)const{
throwAccountException(this,msg);}//以下僅主函數的實現與之前不同,其它皆與之相同intmain(){ Datedate(2008,11,1); //起始日期
Controllercontroller(date); stringcmdLine; constchar*FILE_NAME="commands.txt";
ifstreamfileIn(FILE_NAME); //以讀模式打開文件
if(fileIn) //如果正常打開,就執行文件中的每一條命令 {
while(getline(fileIn,cmdLine)) {
try { controller.runCommand(cmdLine); }catch(exception&e) { cout<<"Badlinein"<<FILE_NAME<<":"<<cmdLine<<endl; cout<<"Error:"<<e.what()<<endl; return1; } } fileIn.close(); //關閉文件
} 綜合實例——對個人銀行賬戶管理程序的改進
ofstreamfileOut(FILE_NAME,ios_base::app); //以追加模式
cout<<"(a)addaccount(d)deposit(w)withdraw(s)show(c)changeday(n)nextmonth(q)query(e)exit"<<endl; while(!controller.isEnd())//從標準輸入讀入命令并執行,直到退出 {
cout<<controller.getDate()<<"\tTotal:"<<Account::getTotal() <<"\tcommand>"; stringcmdLine; getline(cin,cmdLine);
try { if(controller.runCommand(cmdLine)) fileOut<<cmdLine<<endl; //將命令寫入文件
}catch(AccountException&e) { cout<<"Error(#"<<e.getAccount()->getId()<<"):" <<e.what()<<endl; }catch(exception&e) { cout<<"Error:"<<e.what()<<endl; } } return0;}綜合實例——對個人銀行賬戶管理程序的改進運行結果如下:......(前面的輸入和輸出與之前給出的完全相同,篇幅所限,不再重復)2009-1-1Total:20482.9command>w220000buyacarError(#C5392394):notenoughcredit2009-1-1Total:20482.9command>w21500buyatelevision2009-1-1#C5392394-1500-1550buyatelevision2009-1-1Total:18982.9command>q2008-12-52009-1-32Error:Invaliddate2009-1-1Total:18982.9command>q2008-12-52009-1-312008-12-5#S3755217550010500salary2009-1-1#S375521717.7710517.8interest2009-1-1#0234234215.1610015.2interest2009-1-1#C5392394-50-50annualfee2009-1-1#C5392394-1500-1550buyatelevision2009-1-1Total:18982.9command>e異常處理異常處理標準程序庫異常處理綜合實例異常安全性原則智能指針異常安全性問題異常安全性原則一個異常安全的函數,在有異常拋出時:不應泄露任何資源不能使任何對象進入非法狀態反例:第6講中的下列代碼:template<classT,intSIZE>voidStack<T,SIZE>::push(constT&item){ assert(!isFull()); //如果棧滿了,則報錯
list[++top]=item; //將新元素壓入棧頂}如果賦值過程中有異常拋出,由于top已經增1,棧頂的內容將變得不確定異常安全性問題異常安全性原則該函數的修正版本:template<classT,intSIZE>voidStack<T,SIZE>::push(constT&item){ assert(!isFull()); //如果棧滿了,則報錯
list[top+1]=item;//將新元素壓入棧頂
top++;}即使賦值時拋出異常,由于此時top并沒有真正增1,因此當前對象的狀態沒有改變,該函數是異常安全的編寫異常安全程序的原則異常安全性原則明確哪些操作絕對不會拋擲異常這些操作是異常安全編程的基石例:基本數據類型的絕大部分操作,指針的賦值、算術運算和比較運算,STL容器的swap函數盡量確保析構函數不拋擲異常編寫異常安全程序的原則
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯系上傳者。文件的所有權益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
- 4. 未經權益所有人同意不得將文件中的內容挪作商業或盈利用途。
- 5. 人人文庫網僅提供信息存儲空間,僅對用戶上傳內容的表現方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
- 6. 下載文件中如有侵權或不適當內容,請與我們聯系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 出售簡易別墅合同標準文本
- 辦學堂合作合同樣本
- 創業基地勞動合同標準文本
- 中介農村租地合同標準文本
- 供電所光伏合同標準文本
- 中糧貿易購銷合同標準文本
- 農村用房售賣合同標準文本
- 分期購車購車合同樣本
- 利用合同樣本制作
- 公司門衛勞務合同標準文本
- 日常安全生產檢查記錄表格
- 第二講舊石器時代課件
- GB/T 12227-2005通用閥門球墨鑄鐵件技術條件
- GA/T 1769-2021移動警務PKI系統總體技術要求
- 實相般若波羅蜜經
- (更新版)中國移動政企行業認證題庫大全-下(判斷題匯總)
- 上海交通大學醫學院附屬仁濟醫院-日間手術管理信息化實踐與發展
- 2021年湖北理工學院輔導員招聘考試題庫及答案解析
- 消防設備設施維護保養臺賬
- 員工崗位技能考核評定表
- 普通沖床設備日常點檢標準作業指導書
評論
0/150
提交評論