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