學(xué)會(huì)使用C#異常
在 C# 中,程序中在運(yùn)行時(shí)出現(xiàn)的錯(cuò)誤,會(huì)不斷在程序中進(jìn)行傳播,這種機(jī)制稱(chēng)為“異常”。異常通常由錯(cuò)誤的代碼引發(fā),并由能夠更正錯(cuò)誤的代碼進(jìn)行 catch。異常可由 .NET 的 CLR 或由程序中的代碼引發(fā)。一旦引發(fā)了異常,這個(gè)異常將會(huì)在調(diào)用堆棧中一直向上進(jìn)行傳播,直到尋找到跟它匹配的 catch 語(yǔ)句。沒(méi)有 catch 的異常會(huì)由系統(tǒng)提供的默認(rèn)的異常處理程序進(jìn)行處理,也就是你經(jīng)??吹降囊粋€(gè)突然造成調(diào)試中斷并顯示異常信息的對(duì)話框。
所有的異常,它們都是從 Exception 派生出來(lái)的。這些異常的類(lèi)型,都會(huì)包含詳細(xì)描述異常的屬性。在這里我將自定義了一個(gè)新的異常類(lèi),其實(shí)也可以自定義配置異常的屬性(這是可選的),然后我使用 throw 關(guān)鍵字顯示引發(fā)該對(duì)象(即異常)。
/// <summary>
/// 定義新異常
/// </summary>
class MyException : Exception
{
public MyException(string msg) { }
}
/// <summary>
/// 拋出新定義的異常
/// </summary>
static void ThrowMyExcetion()
{
throw new MyException("Sorry, this is test!");
}
在引發(fā)異常之后,運(yùn)行時(shí)程序會(huì)檢查當(dāng)前語(yǔ)句確定它是否包含在 try 塊中。 如果是的話,就會(huì)檢查與該 try 塊相關(guān)聯(lián)的所有 catch 塊,來(lái)確定它們是否能夠 catch 該異常。 catch 塊通常會(huì)指定異常類(lèi)型;如果該 catch 塊的類(lèi)型與異?;蛩幕?lèi)的相同(或匹配),則該 catch 塊就能夠捕獲并處理。
static void Main(string[] args)
{
try
{
ThrowMyExcetion(); //直接調(diào)用拋出異常的方法
}
catch (MyException e)
{
Console.WriteLine(e);
}
Console.Read();
}

如果引發(fā)異常的語(yǔ)句不在 try 塊中,或者包含該語(yǔ)句的 try 塊沒(méi)有匹配的 catch 塊,運(yùn)行時(shí)將檢查調(diào)用方法中是否有 try 語(yǔ)句和 catch 塊。 運(yùn)行時(shí)將在調(diào)用堆棧中繼續(xù)往上搜索兼容(或匹配)的 catch 塊。在找到并執(zhí)行 catch 塊之后,控制權(quán)將傳遞給 catch 塊之后的下一個(gè)語(yǔ)句。
一個(gè) try 語(yǔ)句可能包含多個(gè) catch 塊。 將執(zhí)行第一個(gè)能夠處理該異常的 catch 語(yǔ)句;任何后續(xù)的 catch 語(yǔ)句都將被忽略,即使它們是兼容的也如此。 因此,在任何情況下都應(yīng)該按照從最具體(或者派生程度最高)到最不具體這一順序排列 catch 塊。 例如:
static void Main(string[] args)
{
StreamWriter sw = null;
try
{
sw = new StreamWriter(@"C:\book\小二和小三的故事.txt");
sw.Write("You are 250.");
}
catch (FileNotFoundException e)
{
//將具體的異常放在第一位
Console.WriteLine(e);
}
catch (IOException e)
{
//將并不具體的放在相對(duì)后面的位置
Console.WriteLine(e);
}
catch (Exception e)
{
Console.WriteLine(e);
}
finally
{
if (sw != null)
{
sw.Close();
}
}
Console.Read();
}
執(zhí)行 catch 塊之前,運(yùn)行時(shí)會(huì)檢查 finally 塊。 Finally 塊使程序員能夠清除中止的 try 塊可能遺留下的任何模糊狀態(tài),或者釋放任何外部資源(例如圖形句柄、數(shù)據(jù)庫(kù)連接或文件流),而無(wú)需等待運(yùn)行時(shí)中的垃圾回收器終結(jié)這些對(duì)象。 例如:
static void Main(string[] args)
{
FileStream fs = null;
FileInfo fi = new FileInfo(@"小二和小三的故事.txt");
try
{
fs = fi.OpenWrite();
fs.WriteByte(0);
}
finally
{
//記住哦,如果你忘記 close,將會(huì)引發(fā) IO 異常!
//if (fs != null)
//{
// fs.Close();
//}
}
try
{
fs = fi.OpenWrite();
fs.WriteByte(1);
Console.WriteLine("OK!");
}
catch (IOException e)
{
Console.WriteLine("Fail!");
}
Console.Read();
}

“Fail!”,這是因?yàn)樯厦孀⑨屃诵枰P(guān)閉文件流的語(yǔ)句,你可以嘗試下去掉注釋看看結(jié)果,記住哦,IO 操作都應(yīng)該在結(jié)束時(shí)釋放資源。
如果 WriteByte(0)(第9行) 引發(fā)了異常,那么在沒(méi)有調(diào)用 fs.Close() 的情況下,你在第二個(gè) try 塊中嘗試重新 OpenWrit() 的代碼就會(huì)失敗,因?yàn)榇藭r(shí)文件會(huì)保持鎖定狀態(tài)。 假如你取消注釋?zhuān)捎跁?huì)執(zhí)行 finally 塊(即使已引發(fā)異常),使得可以正確地關(guān)閉文件,從而避免再次引發(fā)異常。
如果在引發(fā)異常之后沒(méi)有在調(diào)用堆棧上找到相匹配的 catch 塊,則會(huì)可能會(huì)出現(xiàn)下面的情況:
- 如果異常出現(xiàn)在析構(gòu)函數(shù)中,則中止該析構(gòu)函數(shù)并調(diào)用基類(lèi)的析構(gòu)函數(shù)(如果有)。
- 如果調(diào)用堆棧包含靜態(tài)構(gòu)造函數(shù)或靜態(tài)字段初始值設(shè)定項(xiàng),則會(huì)引發(fā) TypeInitializationException,并將原始異常分配給新異常的 InnerException 屬性。
- 如果到達(dá)線程的開(kāi)頭,將會(huì)終止線程。
以上就是本文的全部?jī)?nèi)容,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,同時(shí)也希望多多支持腳本之家!
相關(guān)文章
C#關(guān)于System.Collections空間詳解
這篇文章主要介紹了C#關(guān)于System.Collections空間,需要的朋友可以參考下2014-07-07
C#實(shí)現(xiàn)json格式數(shù)據(jù)解析功能的方法詳解
這篇文章主要介紹了C#實(shí)現(xiàn)json格式數(shù)據(jù)解析功能的方法,結(jié)合實(shí)例形式較為詳細(xì)的分析了C#解析json格式數(shù)據(jù)的具體操作步驟與相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-12-12
C#連接Oracle數(shù)據(jù)庫(kù)使用Oracle.ManagedDataAccess.dll
這篇文章主要介紹了C#使用Oracle.ManagedDataAccess.dll的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-11-11
C#中的隊(duì)列Queue<T>與堆棧Stack<T>
這篇文章介紹了C#中的隊(duì)列Queue<T>與堆棧Stack<T>,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05
用C#做網(wǎng)絡(luò)爬蟲(chóng)的步驟教學(xué)
在本篇內(nèi)容里小編給大家分享的是關(guān)于用C#做網(wǎng)絡(luò)爬蟲(chóng)的步驟和方法,需要的朋友們可以參考下。2018-12-12

