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


本篇通过自定义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的异常处理_MVC_02