學會處理異常 你可以使用 try 塊來對你覺得可能會出現異常的代碼進行分區。 其中,與之關聯的 catch 塊可用於處理任何異常情況。 一個包含代碼的 finally 塊,無論 try 塊中是否在運行時引發異常(例如,釋放在 try 塊中分配的資源),這些 finally 塊的代碼都會運行。 這些“ ...
學會處理異常
你可以使用 try 塊來對你覺得可能會出現異常的代碼進行分區。 其中,與之關聯的 catch 塊可用於處理任何異常情況。 一個包含代碼的 finally 塊,無論 try
塊中是否在運行時引發異常(例如,釋放在 try
塊中分配的資源),這些 finally 塊的代碼都會運行。 這些“異常部分”:可以由一個 try
塊、一個或多個關聯的 catch
塊、一個 finally
塊分別組合。
這裡我列舉了 3 種情況:一個 try-catch
語句,一個 try-finally
語句,和一個 try-catch-finally
語句。
try-catch:
1 static void Main(string[] args) 2 { 3 try 4 { 5 //需要執行的代碼 6 } 7 catch (Exception e) 8 { 9 //這裡可以獲取到被捕獲的異常 10 //你需要知道自己應該如何處理該異常 11 } 12 }
try-finally:
1 try 2 { 3 //需要執行的代碼 4 } 5 finally 6 { 7 //在 try 塊後執行的代碼 8 }
try-catch-finally:
1 try 2 { 3 //需要執行的代碼 4 } 5 catch (Exception e) 6 { 7 //這裡處理異常 8 } 9 finally 10 { 11 //在 try 塊(也可能是 catch 塊)後執行的代碼 12 }
【備註】不帶有 catch
或 finally
塊的 try
塊將導致編譯器錯誤。
捕獲異常的 Catch 塊
catch
塊可以指定要捕捉的異常類型,又可以稱為“異常篩選器”。 異常類型都是從 Exception 派生出來。 一般而言,不會將所有異常的基類 System.Exception 指定為要 catch 的“異常篩選器”,除非你非常瞭解如何處理由 try
塊引發的所有異常,或者在 catch
塊中包括了 throw 語句。
多個 catch
塊可以串聯在一起(要求異常篩選器不同)。 多個 catch
塊的執行順序是:在代碼中,從頂部到底部,但是,對於在運行時所引發的每一個異常,程式都只會執行一個 catch
數據塊。 與指定的異常類型或它的基類相匹配的第一個 catch
塊,才會被執行。 通常,我們需要將最特殊(最具體或者說派生程度最最最高)的異常類,這段 catch
塊放在所有 catch 塊的最前面,而他們的基類 Excetion 的 catch 塊就放在最後(當然,也可以不寫)。
在以下條件為真時,你應該選擇 catch 異常:
-
瞭解引發異常的原因,並可實現有選擇性的恢復。例如,在捕獲 FileNotFoundException 時你可以提示用戶“文件找不到”和“請輸入新的文件名”等。
-
你也可以新建一個更具體或者說更具有代表性的異常,並選擇引發該異常。
1 double GetNum(double[] nums,int index) 2 { 3 try 4 { 5 return nums[index]; 6 } 7 catch (IndexOutOfRangeException e) 8 { 9 throw new ArgumentOutOfRangeException("Sorry, 你想要的索引已經超出界限!"); 10 } 11 }
希望在將異常拋出去時,我們通常會選擇處理部分異常。 在下麵這個示例中,catch
塊在再次 throw 異常之前,添加錯誤日誌。
1 try 2 { 3 //嘗試訪問系統資源 4 } 5 catch (Exception e) 6 { 7 //偽代碼:記錄錯誤日誌 8 log.Error(e); 9 10 //再重新拋出錯誤 11 throw; 12 }
釋放資源的 Finally 塊
可以使用 finally
塊釋放(清理)在 try
塊中需要執行釋放(清理)資源的操作。 如果存在finally
塊,它將在最後執行,也就是在 try
塊和任何匹配 catch
塊之後執行。 不管是否引發異常或者說是否找到與異常類型相匹配的 catch
塊,finally
塊它始終都會運行。
可以使用 finally
塊釋放資源(如 IO 流、DB 連接和圖形句柄),而不要等待運行時中的垃圾回收器來完成對象資源的回收。 其實,我們更建議使用 using 語句。
在下麵的示例中,我使用 finally
塊關閉在 try
塊中打開的文件。註意,在關閉文件之前你應該要檢查該文件句柄的狀態。 如果 try
塊無法打開文件,則文件句柄的值依然為 null
,這時, finally
塊就不會嘗試關閉它。 或者說,如果在 try
塊中成功打開該文件,則 finally
塊才會成功地關閉正在打開的文件。
1 static void Main(string[] args) 2 { 3 FileStream fs = null; 4 FileInfo fi = new System.IO.FileInfo("C:\\小二和小三的故事.txt"); 5 6 try 7 { 8 fs = fi.OpenWrite(); 9 fs.WriteByte(0); 10 } 11 finally 12 { 13 // 記得判斷 null 哦,不然可能觸發其它異常 14 if (fs != null) 15 { 16 fs.Close(); 17 } 18 } 19 20 }
C# 基礎回顧系列
《C# 知識回顧 - 表達式樹 Expression Trees》
《C# 知識回顧 - 特性 Attribute》、《剖析 AssemblyInfo.cs - 瞭解常用的特性 Attribute》《C# 知識回顧 - 委托 delegate》、《C# 知識回顧 - 委托 delegate (續)》
《C# 知識回顧 - 事件入門》、《C# 知識回顧 - Event 事件》
《string 與 String,大 S 與小 S 之間沒有什麼不可言說的秘密》
《C# 知識回顧 - 異常介紹》、《C# 知識回顧 - 學會使用異常》
【博主】反骨仔
【原文】http://www.cnblogs.com/liqingwen/p/6193999.html
【參考】微軟官方文檔