詳解C#編程中異常的創(chuàng)建和引發(fā)以及異常處理
創(chuàng)建和引發(fā)異常
異常用于指示在運(yùn)行程序時(shí)發(fā)生了錯(cuò)誤。此時(shí)將創(chuàng)建一個(gè)描述錯(cuò)誤的異常對(duì)象,然后使用 throw 關(guān)鍵字“引發(fā)”該對(duì)象。然后運(yùn)行時(shí)搜索最兼容的異常處理程序。
當(dāng)存在下列一種或多種情況時(shí),程序員應(yīng)引發(fā)異常:
方法無(wú)法完成其中定義的功能。
例如,如果方法的參數(shù)具有無(wú)效值:
static void CopyObject(SampleClass original) { if (original == null) { throw new System.ArgumentException("Parameter cannot be null", "original"); } }
根據(jù)對(duì)象的狀態(tài),對(duì)某個(gè)對(duì)象進(jìn)行不適當(dāng)?shù)恼{(diào)用。
一個(gè)示例可能嘗試對(duì)只讀文件執(zhí)行寫操作。在對(duì)象狀態(tài)不允許某項(xiàng)操作的情況下,引發(fā) InvalidOperationException 的一個(gè)實(shí)例或基于此類的派生類的對(duì)象。以下為引發(fā) InvalidOperationException 對(duì)象的方法的示例:
class ProgramLog { System.IO.FileStream logFile = null; void OpenLog(System.IO.FileInfo fileName, System.IO.FileMode mode) {} void WriteLog() { if (!this.logFile.CanWrite) { throw new System.InvalidOperationException("Logfile cannot be read-only"); } // Else write data to the log and return. } }
方法的參數(shù)導(dǎo)致了異常。
在此情況下,應(yīng)捕獲原始異常并創(chuàng)建一個(gè) ArgumentException 實(shí)例。原始異常應(yīng)作為 InnerException 參數(shù)傳遞給 ArgumentException 的構(gòu)造函數(shù):
static int GetValueFromArray(int[] array, int index) { try { return array[index]; } catch (System.IndexOutOfRangeException ex) { System.ArgumentException argEx = new System.ArgumentException("Index is out of range", "index", ex); throw argEx; } }
異常包含一個(gè)名為 StackTrace 的屬性。此字符串包含當(dāng)前調(diào)用堆棧上的方法的名稱,以及為每個(gè)方法引發(fā)異常的位置(文件名和行號(hào))。 StackTrace 對(duì)象由公共語(yǔ)言運(yùn)行時(shí) (CLR) 從 throw 語(yǔ)句點(diǎn)開(kāi)始自動(dòng)創(chuàng)建,因此必須從堆棧跟蹤的開(kāi)始點(diǎn)引發(fā)異常。
所有異常都包含一個(gè)名為 Message 的屬性。應(yīng)該設(shè)置此字符串來(lái)解釋發(fā)生異常的原因。注意,不應(yīng)將安全敏感信息放在消息文本中。除 Message 之外,ArgumentException 還包含一個(gè)名為 ParamName 的屬性,應(yīng)將該屬性設(shè)置為導(dǎo)致引發(fā)異常的參數(shù)的名稱。對(duì)于屬性設(shè)置器,ParamName 應(yīng)設(shè)置為 value。
公共的受保護(hù)方法應(yīng)在其無(wú)法完成預(yù)期功能時(shí)引發(fā)異常。引發(fā)的異常類應(yīng)該是符合錯(cuò)誤條件的最確切的可用異常。這些異常應(yīng)編寫為類功能的一部分,派生類或?qū)υ碱惖母聭?yīng)保留相同的行為,以實(shí)現(xiàn)向后兼容性。
引發(fā)異常時(shí)要避免的情況
下表確定了在引發(fā)異常時(shí)要避免的做法:
- 不應(yīng)使用異常來(lái)更改正常執(zhí)行過(guò)程中的程序流程。異常只能用于報(bào)告和處理錯(cuò)誤條件。
- 只能引發(fā)異常,而不能作為返回值或參數(shù)返回異常。
- 不要從自己的源代碼中有意引發(fā) System.Exception、System.SystemException、System.NullReferenceException 或 System.IndexOutOfRangeException。
- 不要?jiǎng)?chuàng)建可在調(diào)試模式下引發(fā)但不會(huì)在發(fā)布模式下引發(fā)的異常。若要在開(kāi)發(fā)階段確定運(yùn)行時(shí)錯(cuò)誤,請(qǐng)改用調(diào)試斷言。
定義異常類
程序可以引發(fā) System 命名空間中的預(yù)定義異常類(前面注明的情況除外),或通過(guò)從 Exception 派生來(lái)創(chuàng)建它們自己的異常類。派生類至少應(yīng)定義四個(gè)構(gòu)造函數(shù):一個(gè)是默認(rèn)構(gòu)造函數(shù),一個(gè)用來(lái)設(shè)置消息屬性,一個(gè)用來(lái)設(shè)置 Message 屬性和 InnerException 屬性。第四個(gè)構(gòu)造函數(shù)用于序列化異常。新異常類應(yīng)該可序列化。例如:
public class InvalidDepartmentException : System.Exception { public InvalidDepartmentException() : base() { } public InvalidDepartmentException(string message) : base(message) { } public InvalidDepartmentException(string message, System.Exception inner) : base(message, inner) { } // A constructor is needed for serialization when an // exception propagates from a remoting server to the client. protected InvalidDepartmentException(System.Runtime.Serialization.SerializationInfo info, System.Runtime.Serialization.StreamingContext context) { } }
僅當(dāng)新屬性提供的數(shù)據(jù)有助于解決異常時(shí),才應(yīng)將其添加到異常類。如果向派生的異常類添加了新屬性,則應(yīng)重寫 ToString() 以返回添加的信息。
異常處理
C# 程序員可使用 try 塊對(duì)可能受異常影響的代碼進(jìn)行分區(qū)。關(guān)聯(lián)的 catch 塊用于處理任何結(jié)果異常。一個(gè)包含代碼的 finally 塊,無(wú)論 try 塊中是否引發(fā)異常(例如,釋放在 try 塊中分配的資源),這些代碼都會(huì)運(yùn)行。一個(gè) try 塊需要一個(gè)或多個(gè)關(guān)聯(lián)的 catch 塊或一個(gè) finally 塊,或兩者。
以下示例給出了一個(gè) try-catch 語(yǔ)句,一個(gè) try-finally 語(yǔ)句,和一個(gè) try-catch-finally 語(yǔ)句。
try { // Code to try goes here. } catch (SomeSpecificException ex) { // Code to handle the exception goes here. // Only catch exceptions that you know how to handle. // Never catch base class System.Exception without // rethrowing it at the end of the catch block. } try { // Code to try goes here. } finally { // Code to execute after the try block goes here. } try { // Code to try goes here. } catch (SomeSpecificException ex) { // Code to handle the exception goes here. } finally { // Code to execute after the try (and possibly catch) blocks // goes here. }
不帶有 catch 或 finally 塊的 try 塊將導(dǎo)致編譯器錯(cuò)誤。
Catch 塊
catch 塊可以指定要捕捉的異常的該類型。類型規(guī)范稱為“異常篩選器”。異常類型應(yīng)從 Exception 派生出來(lái)。一般而言,不會(huì)將 Exception 指定為異常篩選器,除非您了解如何處理 try 塊中可能引發(fā)的所有異常,或者您在 catch 塊中包括了 throw 語(yǔ)句。
具有不同異常篩選器的多個(gè) catch 塊可以串聯(lián)在一起。多個(gè) catch 數(shù)據(jù)塊的計(jì)算順序是在代碼中從頂部到底部,但是,對(duì)于所引發(fā)的每個(gè)異常,都只執(zhí)行一個(gè) catch 數(shù)據(jù)塊。與指定的準(zhǔn)確類型或其基類最為匹配的第一個(gè) catch 塊被執(zhí)行。如果 catch 塊沒(méi)有指定匹配異常篩選器,則 catch 塊就不具有選定的篩選器(如果語(yǔ)句有的話)。需要將帶有最具體的(即派生程度最高的)異常類的 catch 塊放在最前面。
當(dāng)下列條件為真時(shí),應(yīng)該捕捉異常:
對(duì)引發(fā)異常的原因有具體的了解,并可實(shí)現(xiàn)特定的恢復(fù),例如,在捕獲 FileNotFoundException 對(duì)象時(shí)提示用戶輸入新的文件名。
可以新建一個(gè)更具體的異常并引發(fā)該異常。
int GetInt(int[] array, int index) { try { return array[index]; } catch(System.IndexOutOfRangeException e) { throw new System.ArgumentOutOfRangeException( "Parameter index is out of range."); } }
希望在將異常傳遞出去進(jìn)行額外處理前部分地處理異常。在下面的示例中,catch 塊用于在再次引發(fā)異常之前,向錯(cuò)誤日志添加條目。
try { // Try to access a resource. } catch (System.UnauthorizedAccessException e) { // Call a custom error logging procedure. LogError(e); // Re-throw the error. throw; }
Finally 塊
可以使用 finally 塊清理在 try 塊中執(zhí)行的操作。如果存在,finally 塊將在最后執(zhí)行,在 try 塊和任何匹配 catch 的塊之后執(zhí)行。不管是否引發(fā)異?;蛘呤欠裾业脚c異常類型匹配的 catch 塊,finally 始終運(yùn)行。
可以使用 finally 塊釋放資源(如文件流、數(shù)據(jù)庫(kù)連接和圖形句柄),而不用等待由運(yùn)行時(shí)中的垃圾回收器來(lái)完成對(duì)象。
在下面的示例中,使用 finally 塊關(guān)閉在 try 塊中打開(kāi)的文件。注意,在關(guān)閉文件之前要檢查該文件句柄的狀態(tài)。如果 try 塊無(wú)法打開(kāi)文件,則文件句柄仍具有值 null,并且 finally 塊不會(huì)嘗試關(guān)閉它?;蛘?,如果在 try 塊中成功打開(kāi)該文件,則 finally 塊將關(guān)閉打開(kāi)的文件。
System.IO.FileStream file = null; System.IO.FileInfo fileinfo = new System.IO.FileInfo("C:\\file.txt"); try { file = fileinfo.OpenWrite(); file.WriteByte(0xF); } finally { // Check for null because OpenWrite might have failed. if (file != null) { file.Close(); } }
相關(guān)文章
C#實(shí)現(xiàn)一個(gè)簡(jiǎn)單實(shí)用的TXT文本操作及日志框架詳解
這篇文章主要給大家介紹了關(guān)于利用C#如何實(shí)現(xiàn)一個(gè)簡(jiǎn)單實(shí)用的TXT文本操作及日志框架的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們一起來(lái)看看吧2018-07-07C#遠(yuǎn)程發(fā)送和接收數(shù)據(jù)流生成圖片的方法
這篇文章主要介紹了C#遠(yuǎn)程發(fā)送和接收數(shù)據(jù)流生成圖片的方法,涉及C#通過(guò)數(shù)據(jù)流傳輸圖片的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-07-07C#使用三層架構(gòu)開(kāi)發(fā)Winform的詳細(xì)案例
這篇文章介紹了C#使用三層架構(gòu)開(kāi)發(fā)Winform的詳細(xì)案例,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-04-04C# 構(gòu)造函數(shù)如何調(diào)用虛方法
這篇文章主要介紹了C# 構(gòu)造函數(shù)如何調(diào)用虛方法,文中講解非常詳細(xì),示例代碼幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下2020-07-07C# 使用multipart form-data方式post數(shù)據(jù)到服務(wù)器
這篇文章主要介紹了C# 使用multipart form-data方式post數(shù)據(jù)到服務(wù)器,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08