1、程序運行時產生的錯誤經過使用一種稱爲異常(Exception)的機制在程序中傳遞,經過異常處理(Exception Handling)有助於處理程序運行過程當中發生的意外或異常狀況;異常可由CLR和客戶端代碼拋出(Throw),拋出的異常會在調用堆棧中傳遞,直到遇到能夠捕獲該異常的語句進行處理並停止傳遞,未捕獲的異常會由系統終止進程並由系統的通用異常處理程序處理,一般會顯示一個包含異常信息的對話框;服務器
1.異常是經過一個特殊類型的對象進行傳遞的,其基類是位於命名空間System中的類Exception,自定義的異常類型必須繼承自該類並以Exception結尾,異常對象中包含描述異常的自定義數據:函數
public class MyException : Exception { public string MyInfo; }
※基類Exception中的屬性Message包含拋出異常的緣由,屬性StackTrace包含拋出異常時調用堆棧上方法的名稱、位置和行號,此屬性由CLR在throw語句的調用位置自動建立,重寫的ToString()方法會打印異常的類型名稱、屬性Message和StackTrace,只讀屬性InnerException一般用來保存拋出異常的原始異常,須要在新建立異常對象時經過構造函數傳入;spa
※完整的自定義異常類型應添加可序列化特性Serializable並至少聲明四個構造函數:默認構造函數、設置Message屬性的構造函數、設置Message和InnerException屬性的構造函數、用於序列化異常的構造函數:線程
using System.Runtime.Serialization; [Serializable] public class MyException : Exception { public MyException() : base() { } public MyException(string message) : base(message) { } public MyException(string message, Exception inner) : base(message, inner) { } //當異常從遠程處理服務器傳播到客戶端時,須要該構造函數進行序列化 protected MyException(SerializationInfo info, StreamingContext context) : base(info, context) { } }
2.拋出異常時會涉及異常類型對象的實例化和初始化,並經過關鍵字throw拋出該異常:code
public static void MyFunc() { throw new MyException { MyInfo = "My Exception Throw." }; }
3.使用try語句塊對可能拋出異常的代碼進行包裹,使用關聯的catch語句塊處理拋出的異常,使用關聯的finally語句塊聲明不管是否拋出異常都會執行的語句,例如釋放try語句塊中分配的非託管資源;完整的異常處理語句須要包含一個try語句塊和:一個或多個catch語句塊、一個finally語句塊或兩者都有,即try-catch-(finally)、try-finally;orm
※一個try語句塊能夠包含多個catch語句塊,catch語句塊用來指定要捕獲異常的類型,也被稱爲異常篩選器,catch語句也能夠不指定篩選器,此時默認將基類Object做爲篩選器,通常狀況下必須指定篩選器且不要直接指定基類Exception做爲篩選器,除非瞭解可能拋出的全部類型的異常或在catch語句塊末尾使用throw從新拋出該異常(此時將調用rethrow指令);catch語句應該在理解異常拋出的緣由並能夠實現特定的恢復時才使用,也能夠用來進行部分處理並從新拋出或拋出一個更加具體的異常;catch語句塊應該按照派生程度由高到底的順序聲明:
t對象
ry { MyFunc(); } catch (MyException e) //能夠捕獲MeException類型的異常,若是不使用異常對象,變量e能夠省略 { Console.WriteLine("MyException Throw:" + e.MyInfo); //throw new MyException { MyInfo = "Detailed Information." }; } catch (Exception e) //能夠捕獲全部類型的異常 { Console.WriteLine("Unkown Exception:" + e); throw; //只有在catch語句塊中能夠這麼使用 } //catch //一般用於捕獲非CLS異常 //{ //} finally { //do… }
4.拋出異常時,CLR會在調用堆棧中按順序搜索當前異常對象所兼容的異常篩選器(當前類型或其基類),當找到第一個兼容的異常篩選器後再也不搜索其它的異常篩選器(所以應該將派生程度最高的catch塊放在最前),此時會按照原調用堆棧的順序執行finally語句塊,而後調用捕獲到異常的catch語句塊,而後調用其finally語句塊,最後將控制流傳遞給該語句塊的下一語句繼續執行;blog
※若是拋出的異常在調用堆棧中沒有找到兼容的異常篩選器,會出現如下三種狀況:
1.若是異常在析構函數中拋出,則將停止該析構函數,並調用基類析構函數(若是有);
2.若是調用堆棧中包含靜態構造函數或靜態字段初始化,則將拋出異常TypeInitializationException,並將原異常分配給其屬性InnerException;且在當前應用的生命週期內CLR不會再次調用該函數,未執行的代碼塊將永遠不會執行;
3.若是異常到達線程的開頭,則將終止該線程,並由系統終止進程並處理;
繼承
若是您以爲閱讀本文對您有幫助,請點一下「推薦」按鈕,您的承認是我寫做的最大動力!生命週期
做者:Minotauros
出處:https://www.cnblogs.com/minotauros/
本文版權歸做者和博客園共有,歡迎轉載,但未經做者贊成必須保留此段聲明,且在文章頁面明顯位置給出原文鏈接,不然保留追究法律責任的權利。