集成EntLib实现ASP.NET MVC的异常处理

本篇通过自定义ASP.NET MVC的异常筛选器实现了与EntLib的EHAB(Exception Handling Application Block)的集成,使我们可以通过配置的方式来定义异常处理策略,并最终通过错误页面显示被处理过的异常信息。[源代码从这里下载]

我们知道ASP.NET MVC具有一个类型为HandleErrorAttribute的异常筛选器可以起到错误页面的导向作用。在这里我直接让我们自定义的异常筛选器继承它,为此我们定义了如下一个名称为ExtendedHandleErrorAttribute的类型。我们通过指定异常处理策略的配置名称来创建ExtendedHandleErrorAttribute,而属性ExceptionPolicy则表示具体的异常处理策略。在重写的OnException方法中,我们在try/catch中调用了ExceptionPolicyImpl的HandleException方法,而传入该方法的对象为需要处理的异常。捕获的异常可能是原来的异常,也可能是处理后的异常,这依赖于postHandlingAction的设置。

   1: public class ExceptionHandlingAttribute: HandleErrorAttribute
   2: {
   3:     public ExceptionPolicyImpl ExceptionPolicy { get; private set; }    
   4:  
   5:     public ExceptionHandlingAttribute(string exceptionPolicyName)
   6:     {
   7:         this.ExceptionPolicy = EnterpriseLibraryContainer.Current.GetInstance<ExceptionPolicyImpl>(exceptionPolicyName);
   8:     }
   9:  
  10:     public override void OnException(ExceptionContext filterContext)
  11:     {
  12:         try
  13:         {
  14:             this.ExceptionPolicy.HandleException(filterContext.Exception);
  15:         }
  16:         catch (Exception ex)
  17:         {
  18:             filterContext.Exception = ex;
  19:             base.OnException(filterContext);
  20:         }
  21:     }
  22: }

接下来我们来定义显示错误信息的View。我们在Views\Shared目录下创建一个Model类型为HandleErrorInfo的Error.cshtml文件,下面是整个文件的内容。从中可以看出,我们显示了异常的消息、类型和堆栈追踪信息。

   1: @model System.Web.Mvc.HandleErrorInfo
   2: @{
   3:     ViewBag.Title = "Error";
   4: }
   5: <h2>@this.Model.Exception.Message</h2>
   6: <p><b>Exception Type: </b>@this.Model.Exception.GetType()</p>
   7: <p><b>StackTrace: </b>@this.Model.Exception.StackTrace</p>

然后我们如下一个名称为HomeController的控制器,在Action方法Index中,我们执行一个被除数为零的整形除法运算让它抛出DivideByZeroException异常。而我们自定义的异常筛选器直接应用在了HomeController类型上,指定异常处理策略名称为UI Policy,View属性被设置为上面创建的用于显示错误信息的View名称。

   1: [ExtendedHandleError("UI Policy", View = "Error")]
   2: public class HomeController : Controller
   3: {
   4:     public ActionResult Index()
   5:     {
   6:         int x = 1;
   7:         int y = 0;
   8:         int result = x / y;
   9:         return View();
  10:     }
  11: }

最后来看定义的Web.config中的异常处理策略,针对抛出的DivideByZeroException异常,我们将其替换成了CalculationErrorException异常,并指定了被替换后的异常消息为”Calculation Error…”。至于PostHandlingAction属性,则被设置为ThrowNewException,意味着被处理后的异常会被抛出来。对了我们的例子来说,也就是说被替换后的CalculationErrorException会被抛出。

   1: <configuration>
   2:   <configSections>
   3:     <section name="exceptionHandling" 
   4:              type="Microsoft.Practices.EnterpriseLibrary.ExceptionHandling.Configuration.ExceptionHandlingSettings, Microsoft.Practices.EnterpriseLibrary.ExceptionHandling"/>
   5:   </configSections>
   6:   ...
   7:   <exceptionHandling>
   8:     <exceptionPolicies>
   9:       <add name="UI Policy">
  10:         <exceptionTypes>
  11:           <add name="InvalidOperationException"
  12:                type="System.DivideByZeroException, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089"
  13:                postHandlingAction="ThrowNewException" >
  14:             <exceptionHandlers>
  15:               <add name="ReplaceHandler"                   
  16:                    type="Microsoft.Practices.EnterpriseLibrary.ExceptionHandling.ReplaceHandler,Microsoft.Practices.EnterpriseLibrary.ExceptionHandling, Version=5.0.414.0, Culture=neutral, PublicKeyToken=a20767533a162583"
  17:                    replaceExceptionType="Artech.Web.Mvc.Extensions.CalculationException, EhabIntegration"
  18:                    exceptionMessage="Calculation Error..."/>
  19:             </exceptionHandlers>
  20:           </add>
  21:         </exceptionTypes>
  22:       </add>
  23:     </exceptionPolicies>
  24:   </exceptionHandling>
  25: </configuration>

我们现在来运行我们的程序,由于HomeController和Index为默认的控制器和Action,所以直接就会导向到出错界面,并显示我们替换后的异常信息。

集成EntLib实现ASP.NET MVC的异常处理


作者:蒋金楠
微信公众账号:大内老A
微博:www.weibo.com/artech
如果你想及时得到个人撰写文章以及著作的消息推送,或者想看看个人推荐的技术资料,可以扫描左边二维码(或者长按识别二维码)关注个人公众号(原来公众帐号蒋金楠的自媒体将会停用)。
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
上一篇:python编程-29:Scrapy爬虫基本使用


下一篇:Linux下telnet允许root用户登录