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

image

posted @ 2011-12-13 13:28 Artech 阅读(1962) 评论(14) 编辑 收藏

等待购买你的第2本书呢。
 回复 引用 查看   
#2楼[楼主] 2011-12-13 13:35 Artech      
引用C#通用权限管理系统组件:等待购买你的第2本书呢。

快了:)

 回复 引用 查看   
#3楼 2011-12-13 13:44 .Robot      
我们是重写Controller的OnException
protected override void OnException(ExceptionContext filterContext)
        {
}


没有用Filter,不知两者区别何在?

 回复 引用 查看   
#4楼 2011-12-13 13:46 .Robot      
public class BaseController : Controller
{
protected override void OnException(ExceptionContext filterContext)
{
}
}


我们定义了一个BaseController,所有的Controller都集成这个BaseController。

 回复 引用 查看   
#5楼[楼主] 2011-12-13 13:52 Artech      
引用.Robot:
我们是重写Controller的OnException
protected override void OnException(ExceptionContext filterContext)
        {
}


没有用Filter,不知两者区别何在?

没有本质的区别,因为Controller本身是一个Filter
public abstract class Controller : ControllerBase, IActionFilter, IAuthorizationFilter, IDisposable, IExceptionFilter, IResultFilter


 回复 引用 查看   
#6楼[楼主] 2011-12-13 13:54 Artech      
引用.Robot:
public class BaseController : Controller
{
protected override void OnException(ExceptionContext filterContext)
{
}
}


我们定义了一个BaseController,所有的Controller都集成这个BaseController。

解决问题的两种方式罢了。

 回复 引用 查看   
#7楼 2011-12-13 13:57 .Robot      
OK.THANK YOU.
 回复 引用 查看   
#8楼 2011-12-13 14:20 Paypal顾问      
好的
 回复 引用 查看   
#9楼 2011-12-13 14:44 超级奶崽      
引用Artech:
引用C#通用权限管理系统组件:等待购买你的第2本书呢。

快了:)

您的下一本书还是写wcf的么

 回复 引用 查看   
#10楼[楼主] 2011-12-13 15:04 Artech      
引用超级奶崽:
引用Artech:
引用C#通用权限管理系统组件:等待购买你的第2本书呢。

快了:)

您的下一本书还是写wcf的么

《WCF技术剖析(卷2)》

 回复 引用 查看   
#11楼 2011-12-13 16:24 artwl      
嗯,不错
 回复 引用 查看   
#12楼 2011-12-13 19:04 Mr Yang      
Good!支持啊!
 回复 引用 查看   
#13楼 2011-12-14 22:12 八位机      
如果是ajax异常,有啥处理实践不?
 回复 引用 查看   
#14楼 2011-12-21 13:22 .Robot      
如果是ajax异常,有啥处理实践不?
我也想知道。

发表评论

昵称: [登录] [注册]

主页:

邮箱:(仅博主可见)

评论内容:

  登录  注册

[使用Ctrl+Enter键快速提交评论]

0 2285966 WXUnlEAS0hk=