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

来源:岁月联盟 编辑:exp 时间:2011-12-14

 

本篇通过自定义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,所以直接就会导向到出错界面,并显示我们替换后的异常信息。

 

image

 

作者:Artech