一些.NET對(duì)多線程異常處理技巧分享
多線程環(huán)境
在我們的產(chǎn)品 SE 中,出現(xiàn)多線程的地方主要有兩大類,一類是通過 ThreadPool 或 new Thread 主動(dòng)發(fā)起多線程,另一類是 Socket 通訊回調(diào)。
多線程異常捕獲
對(duì)于一般的異常處理來說,我們只要簡(jiǎn)單的將可能出錯(cuò)的語句包含在 try/catch 語句中即可。我也曾經(jīng)簡(jiǎn)單的將該方法運(yùn)用于多線程的異常捕獲,結(jié)果并非如此,代碼如下:
public static void Main()
{
try
{
new Thread (Go).Start();
}
catch (Exception ex)
{
// 永遠(yuǎn)執(zhí)行不到這兒!
Console.WriteLine ("Exception!");
}
}
private static void Go()
{
throw null;
}
正確的做法應(yīng)該是在新線程入口方法 Go 中捕獲異常:
public static void Main()
{
new Thread (Go).Start();
}
private static void Go()
{
try
{
...
throw null; // 該異常將會(huì)被捕獲
...
}
catch (Exception ex)
{
// 異常日志記錄,或者通知其他線程出現(xiàn)異常了
...
}
}
以上的正確做法來自 Threading in C# 中的小節(jié) Exception Handling,該文涉及到 .NET 多線程的方方面面,是我看到最全最好的文章。
正確捕獲多線程異常的方法找到了,接下來我們自然會(huì)想:是不是每個(gè)線程入口方法都得這么做?
且看 Threading in C# 中的小節(jié) Exception Handling 的描述:從 .NET 2.0 開始,任何一個(gè)線程上未處理的異常都會(huì)導(dǎo)致整個(gè)應(yīng)用程序關(guān)閉。因此,在每個(gè)線程入口方法中都必須要使用 try/catch 語句,至少在產(chǎn)品應(yīng)用程序中必須如此,以免應(yīng)用程序因?yàn)槲覀兾搭A(yù)料到的代碼而關(guān)閉整個(gè)應(yīng)用程序。
如果僅僅記下異常信息而不在乎應(yīng)用程序異常關(guān)閉,那么有兩個(gè)方法可以做到:
1、對(duì)于 Windows Form 程序來說,有一個(gè)全局異常處理事件:Application.ThreadException;
2、對(duì)于所有 .NET 程序來說,還有一個(gè)更低級(jí)的全局異常處理事件:AppDomain.UnhandledException;
更高的要求
我們能簡(jiǎn)單的通過全局異常處理事件來記錄錯(cuò)誤日志;如果保證不中斷應(yīng)用程序,也可以在每個(gè)線程入口方法中捕獲異常并記錄異常日志。有沒有辦法做到:既能捕獲異常且不中斷應(yīng)用程序,又能如全局異常處理事件那樣簡(jiǎn)單捕獲異常?
對(duì)于主動(dòng)創(chuàng)建的新線程,至少可以做到這一點(diǎn):
public static class ThreadExecutor
{
public static bool Execute(System.Threading.WaitCallback callback, object state)
{
try
{
return System.Threading.ThreadPool.QueueUserWorkItem(callback, state);
}
catch (Exception e)
{
// log the exception
}
return false;
}
}
相關(guān)文章
在SQL Server中使用CLR調(diào)用.NET方法實(shí)現(xiàn)思路
在.NET中新建一個(gè)類,并在這個(gè)類里新建一個(gè)方法,然后在SQL Server中調(diào)用這個(gè)方法,接下來我們將實(shí)現(xiàn)這個(gè)功能做了以下幾個(gè)步驟,詳細(xì)看下本文,感興趣的你可不要錯(cuò)過了哈2013-02-02asp.net 在處理向該請(qǐng)求提供服務(wù)所需的配置文件時(shí)出錯(cuò)
遭遇:“說明: 在處理向該請(qǐng)求提供服務(wù)所需的配置文件時(shí)出錯(cuò)。請(qǐng)檢查下面的特定錯(cuò)誤詳細(xì)信息并適當(dāng)?shù)匦薷呐渲梦募??!卞e(cuò)誤2010-03-03ASP.NET中利用DataList實(shí)現(xiàn)圖片無縫滾動(dòng) 實(shí)例分享
這個(gè)問題之前也困擾我,后來解決了,拿出來分享下,以后用也方便,代碼很容易看懂,不多說什么了2013-06-06自寫一個(gè)模仿Dictionary與Foreach的實(shí)現(xiàn)及心得總結(jié)
利用閑暇時(shí)間自己寫一個(gè)類模仿Dictionary實(shí)現(xiàn),如果一個(gè)類進(jìn)行foreach的話,該類必須實(shí)現(xiàn)IEnumerable,集合要支持foreach方式的遍歷,必須實(shí)現(xiàn)IEnumerable接口,感興趣的你可不要錯(cuò)過了哈2013-02-02URL中去除指定參數(shù)實(shí)現(xiàn)C#代碼
URL中去除指定參數(shù)在項(xiàng)目開發(fā)中還是很常見的,本文將介紹下它在c#代碼中的實(shí)現(xiàn),感興趣的朋友可以參考下哈2013-04-04JavaScript用JQuery呼叫Server端方法實(shí)現(xiàn)代碼與參考語法
從Javascript客戶端用JQuery呼叫Server端的方法,這也是一個(gè)大膽的嘗試,本人做了演示動(dòng)畫以及參考語法,感興趣的朋友可以參考下,希望本人對(duì)你有所幫助2013-01-01asp.net 2.0 中的URL重寫以及urlMappings問題
asp.net 2.0 中的URL重寫以及urlMappings問題...2007-04-04asp.net 實(shí)現(xiàn)自定義Hashtable (.net)
asp.net Hashtable自定義實(shí)現(xiàn)代碼,比較多,大家可以看下,測(cè)試。2009-06-06Asp.net基于ajax和jquery-ui實(shí)現(xiàn)進(jìn)度條
這篇文章主要介紹了Asp.net基于ajax和jquery-ui實(shí)現(xiàn)進(jìn)度條,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-12-12一個(gè)完整的ASP.NET 2.0 URL重寫方案[翻譯]
這篇文章描述了一個(gè)完整的 ASP.NET 2.0 URL 重寫方案。這個(gè)方案使用正則表達(dá)式來定義重寫規(guī)則并解決通過虛擬 URLs 訪問頁面產(chǎn)生回發(fā)事件的一些可能的困難。2009-11-11