前言:在实际开发过程中,对于一个应用系统来说,应该有自己的一套成熟的异常处理框架,这样当异常发生时,也能得到统一的处理风格,将异常信息优雅地反馈给开发人员和用户。我们都知道,.net的异常处理是按照“异常链”的方式从底层向高层逐层抛出,如果不能尽可能地早判断异常发生的边界并捕获异常,CLR会自动帮我们处理,但是这样系统的开销是非常大的,所以异常处理的一个重要原则是“早发现早抛出早处理”。但是本文总结的服务端公共捕获异常处理可以宽泛地看做是在表现层的操作,要捕获特定层的特定异常,不在讨论范围内。
1、BasePage类处理方式
在页面的公共基类里重写OnError事件。在前面这篇《asp.net实现非常实用的自定义页面基类》里,楼猪已经贴了代码,就不再费事了。根据经验,很多人开发的时候几乎都这么写,而且对调试和维护还是很有帮助的。需要说明的是,每新添一个页面,其对应类都必须继承自BasePage类异常处理才起作用。
2、Global.asax处理方式
如1中所述,BasePage类的异常处理要求每一个aspx类文件都继承它,适用性和性能显然会打折扣。而Global.asax文件定义了asp.net应用程序中的所有应用程序对象共有的方法、属性和事件,我们可以不采用BasePage的处理方式,在Global.asax里实现Application_Error事件并处理也可以。下面模仿BasePage类里的处理异常方法,实现如下:
/// <summary> /// 出错处理:写日志,导航到公共出错页面 /// </summary> /// <param></param> /// <param></param> protected void Application_Error(object sender, EventArgs e) { if (Server.GetLastError() == null) return; Exception ex = Server.GetLastError().GetBaseException(); string error = this.DealException(ex); DotNet.Common.Util.Logger.WriteFileLog(error, HttpContext.Current.Request.PhysicalApplicationPath + "LogFile"); if (ex.InnerException != null) { error = this.DealException(ex); DotNet.Common.Util.Logger.WriteFileLog(error, HttpContext.Current.Request.PhysicalApplicationPath + "LogFile"); } this.Server.ClearError(); this.Response.Redirect("/Error.aspx"); } /// <summary> /// 处理异常,用来将主要异常信息写入文本日志 /// </summary> /// <param></param> /// <returns></returns> private string DealException(Exception ex) { this.Application["StackTrace"] = ex.StackTrace; this.Application["MessageError"] = ex.Message; this.Application["SourceError"] = ex.Source; this.Application["TargetSite"] = ex.TargetSite.ToString(); string error = string.Format("URl:{0}\n引发异常的方法:{1}\n错误信息:{2}\n错误堆栈:{3}\n", this.Request.RawUrl, ex.TargetSite, ex.Message, ex.StackTrace); return error; }
上面方式的好处是,写一次代码,应用程序发生的大部分异常它都给你捕捉处理了。楼猪要在这里由衷地发一番感慨,感谢ms为我们提供了这么优秀的框架,太省事了吧。
3、IHttpModule接口处理
1和2的处理方式大家都是非常熟悉的,楼猪在实际开发中基本上都是遵循上面两种写法,而且楼猪因为有了2中这种大小通吃的处理方式,甚至已经激动地感谢ms了。但是,在asp.net程序调用线程进行异步处理的时候,容易发生在后台线程或线程池里抛出的异常并不能被1或(和)2完全捕捉到,这就涉及到asp.net下未捕获异常的处理。也就是说楼猪以前做过的很多大小项目中对异常的处理是不完备的。这难道是nc楼猪没有先谢国家种下的恶果吗?感谢国家,感谢ms,感谢博客园,感谢无私的xdjm,感谢自己......
asp.net下未捕获异常的处理步骤如下:
(1)、创建一个实现IHttpModule接口的类
using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.UI; using System.Web.UI.WebControls; using System.Text; namespace DotNet.Common.WebForm { using DotNet.Common.Util; /// <summary> /// 通用未捕获异常处理 /// </summary> public class AspNetUnhandledExceptionModule : IHttpModule { static object syncObj = new object(); static bool isInit = false; public AspNetUnhandledExceptionModule() { } #region IHttpModule Methods public void Init(HttpApplication context) { lock (syncObj) { if (!isInit) { AppDomain.CurrentDomain.UnhandledException += new UnhandledExceptionEventHandler(OnUnhandledException); isInit = true; } } } public void Dispose() { } #endregion #region OnUnhandledException void OnUnhandledException(object o, UnhandledExceptionEventArgs e) { if (e.ExceptionObject == null) return; Exception ex = e.ExceptionObject as Exception; string error = string.Format("引发异常的方法:{0}\n错误信息:{1}\n错误堆栈:{2}\n", ex.TargetSite, ex.Message, ex.StackTrace); Logger.WriteFileLog(error, AppDomain.CurrentDomain.BaseDirectory + "LogFile"); } #endregion } }
(2)、web.config节点配置
<httpModules> <add type="DotNet.Common.WebForm.AspNetUnhandledExceptionModule, DotNet.Common.WebForm"></add> </httpModules>
最后贴出测试代码: