1、程序中使用try catch
对于预知会发生异常的代码段使用try catch主动捕获异常,适用于提示给用户或跳转到错误页面,或者通过其它方式处理异常(日志、通知等)。
int i = 10; int j = 0; try { Label1.Text = (i / j).ToString(); } catch (Exception ex) { // 这里处理异常:Redirect、Transfer、Log、Notice等 Console.WriteLine("Page:" + ex.Message); }
2、Global中使用Application_Error
如果异常在程序中没有被处理(如没有try catch),则异常的处理会流转到这个方法,这里边可以对异常进行处理。但是此方式不能捕捉子线程中的异常。
int i = 10; int j = 0; Label2.Text = (i / j).ToString();
void Application_Error(object sender, EventArgs e) { // 在出现未处理的错误时运行的代码 Server.Transfer("ErrorPage.aspx"); }
string message = HttpContext.Current.Error != null ? (HttpContext.Current.Error.InnerException != null ? HttpContext.Current.Error.InnerException.Message : string.Empty) : string.Empty; Label1.Text = message;
3、在web.config中配置
出现错误后跳转到ErrorPage.aspx,和Application_Error类似,采用redirectMode模式可以传递异常到错误页面。
4、使用FirstChance异常通知。
关联到AppDomain,如果应用程序域内发生异常,则会首先触发这个事件,然后才查找catch块处理异常。不过在这个事件中不能处理异常,不能消灭异常,只是可以按照通知进行处理。因为如果这里处理了异常,catch块就不能进行处理了。
void Application_Start(object sender, EventArgs e) { // 在应用程序启动时运行的代码 AppDomain.CurrentDomain.FirstChanceException += new EventHandler<System.Runtime.ExceptionServices.FirstChanceExceptionEventArgs>(CurrentDomain_FirstChanceException); } void CurrentDomain_FirstChanceException(object sender, System.Runtime.ExceptionServices.FirstChanceExceptionEventArgs e) { Console.WriteLine(e.Exception.Message); // 错误:响应在上下文中不能使用 // Response.Redirect("ErrorPage.aspx"); // 错误:未将对象引用设置到对象的实例 // Server.Transfer("ErrorPage.aspx"); }
5、绑定UnhandledException事件
关联到AppDomain,关于这个事件并不是每次都能触发,和使用的方式有关,情况比较多。一般情况下我们只能获取这个异常,而不能阻止中断应用程序。
下边给出一个例子:
void Application_Start(object sender, EventArgs e) { // 在应用程序启动时运行的代码 AppDomain.CurrentDomain.UnhandledException += new UnhandledExceptionEventHandler(CurrentDomain_UnhandledException); } void CurrentDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e) { Console.WriteLine(e.Exception.Message); }
一般的未处理异常都会被Application_Error捕捉到,我们这里在线程中抛出一个异常。
另外StackOverflowException在.net4中不再能被UnhandledException捕捉到。
private void CutString() { //throw (new Exception("Test Unhandled exception")); //throw (new StackOverflowException("Test Unhandled exception")); } protected void Button1_Click(object sender, EventArgs e) { Thread th = new Thread(new ThreadStart(CutString)); th.Start(); }
更多请参考:
http://msdn.microsoft.com/zh-Cn/library/system.appdomain.unhandledexception.aspx
http://mlichtenberg.wordpress.com/2011/09/19/catching-unhandled-exceptions-in-asp-net/
6、延伸:子线程异常的处理。网上有介绍通过代理在主线程处理子线程的异常,但是在asp.net中是无状态的,主线程有可能很快消失,其中的某些处理可能执行失败。
这里使用Thread.Sleep使主线程不会很快结束。这种异常处理起来很麻烦,不建议在asp.net中使用处理时间很长的线程。
protected void Button1_Click(object sender, EventArgs e) { Thread th = new Thread(new ThreadStart(() => { try { throw (new Exception("Test Unhandled exception")); } catch (Exception ex) { //跳转到错误页面 Response.Redirect("ErrorPage.aspx"); } })); th.Start(); // asp.net主线程会很快结束,这里让他等等页面跳转。 Thread.Sleep(2000); }
本文列举了处理异常的几种方式,有通过订阅AppDomain事件的方式,有通过配置文件的方式,还有通过Global的方式,最后还对子线程异常的处理谈了一点想法,但是都没有提供一个完善的解决方案,有兴趣的朋友可以自己试试。
个人独立博客:http://blog.bossma.cn/dotnet/asp-net-resolve-exception-some-methods/