asp.net MVC3 仿照博客园功能 异常处理

这几天感觉我自己比较懒,除了偶尔的写了一点博客以外,对于那个重要的仿照博客园功能的项目我竟然忘记更新了。失误了,天天脑子里的事太多了,有点乱,程序员,伤不起啊。

还是不能不谦虚啊,今天继续项目的开发,不做那些费时的工作了,那个等我星期休息的时候再做,今天来做一个异常处理的功能。
    
众所周知,在asp.net世界中,对于异常的处理就是try catch,当然这个是在方法内部,还有Page_Error 页面级别的错误,

application_error 应用程序级别的错误,这几个地方可以用来处理异常信息。一般情况下,

我们对于已知的可能出现的错误会直接在try  catch中捕获并且处理,但是对于那些我们没有考虑到的异常信息,我们就要采取别的办法了,

比如定义一个基类,用来处理子类抛出的异常,然后在可能出现异常信息的地方继承该类,让父类处理子类中不能确定的并且是抛出的未处理的异常信息。

这个和验证用户是否登陆的方式基本一样,对于某些资源,需要有权限的用户才能访问,我们就会定义一个基类,用来验证用户是否登陆,

如果登陆才可以继续执行,否则就要跳转到登陆界面。

当然,前面说的处理异常的方式是在web form时代我们(起码是我)常用的方式,但是在MVC 时代到临的时候,这些处理是否更容易呢?

答案是肯定的,因为技术是在不断进步的嘛。

在MVC中我们对于已知的异常信息仍然会采用try catch的方式,并且是很常用的方式,但是对于验证用户登陆或者catch未捕获的异常信息

(当然我的意思是catch(exception ex)   在catch最后没有这一条,或者说有这一句,但是把异常信息进行了抛出处理),我们有了更容易实现的方式。

下面让我们一起走进如何使用MVC处理异常信息。

在软件架构不断发展的同时,设计模式也有了很多变种,AOP(面向切面编程或面向方面编程)就是一个变种的设计模式,对于设计模式的学习,

我个人不推荐死记硬背,我希望可以了解每种模式的意义,然后在项目中不要刻意的去使用它,而是在重构的时候进行,

这样我们可以更加深入的了解设计模式背后包含的含义。举个例子,单例模式是一个经典的设计模式,它可以保证类的实例只有一个,

但是如果我们不正确的使用这种方式,有时候会带来负面效果,比如多线程同时访问,只有在保证加锁、解锁的情况下可以保证,

但是在普通情况下就会出现意想不到的错误信息。

面向对象编程具有继承性,这应该是面向对象三大特性之一,这是类的垂直方面的编程工作,有上下级或父子关系,AOP是水平方面的编程,

它可以保证在开始之前或结束之后进行,不会破坏里面的结构。个人粗浅理解。

在MVC 3中对AOP的支持就表现在filter  过滤器上,他可以保证在开始之前或结束之后进行。

对于异常的处理我们采用的是自定义异常处理信息继承自IExceptionFilter,当然在MVC总内置了一个HandleErrorAttribute也可以用来捕获异常,

但是我们自己来控制可能会更好一些。

 

首先我们在HomeController的Index方法中抛出一个异常信息

 1  public ActionResult Index(int? id, int pageSize = 20)
 2         {
 3             List<BlogInfo> blogList = BlogServices.GetAllBlogList().ToPagedList<BlogInfo>(id == null ? 1 : Convert.ToInt32(id), pageSize);
 4             ViewBag.BlogList = blogList;
 5             if (Request.IsAjaxRequest())
 6             {
 7 
 8             }
 9             
10                 throw new HttpException(500, "");//抛出异常信息
11             
12 
13             return View(blogList);
14         }
15   //404Not Found
16         public ActionResult NotFound()
17         {
18             return View();
19         }
20         //内部服务器错误 500
21         public ActionResult InternalError()
22         {
23             return View();
24         }

2.设置自定义处理异常类

 1  public class CustomExceptionAttribute :FilterAttribute,IExceptionFilter   //HandleErrorAttribute
 2     {
 3 
 4         public  void OnException(ExceptionContext filterContext)
 5         {
 6             if (filterContext.ExceptionHandled == true)
 7             {
 8                 HttpException httpExce = filterContext.Exception as HttpException;
 9                 if (httpExce.GetHttpCode() != 500)//为什么要特别强调500 因为MVC处理HttpException的时候,如果为500 则会自动
10                     //将其ExceptionHandled设置为true,那么我们就无法捕获异常
11                 {
12                     return;
13                 }
14             }
15             HttpException httpException = filterContext.Exception as HttpException;
16             if (httpException != null)
17             {
18                 filterContext.Controller.ViewBag.UrlRefer = filterContext.HttpContext.Request.UrlReferrer;
19                 if (httpException.GetHttpCode() == 404)
20                 {
21                     filterContext.HttpContext.Response.Redirect("~/home/notfound");
22                 }
23                 else if (httpException.GetHttpCode() == 500)
24                 {
25                     filterContext.HttpContext.Response.Redirect("~/home/internalError");
26                 }
27             }
28             //写入日志 记录
29             filterContext.ExceptionHandled = true;//设置异常已经处理
30         }
31     }
在这里我要多说一句,有些园友可能遇到这么一个问题,就是抛出了一个500的内部服务器异常,但是在自定义异常信息中就是无法捕获到,是什么原因呢?

其实就是MVC对500 的特殊照顾,在HttpException的httpCode为500的时候,MVC框架会自动的处理,然后将其ExceptionHandled设置为true。

对于其他的异常状态码,比如404就没有这样的照顾,所以500我们要优先照顾呀。

3.在设置好了自定义异常处理以后,我们可以在每个Action或Controller中进行注入,但是在MVC 3 中提供了另外一种可以全局注入的方式,那就是全局Filter,我一般叫做全局筛选器。这样就相当于我们在所有的Action上都进行了注入。

在global.asax中,进行全局注册
1  public static void RegisterGlobalFilters(GlobalFilterCollection filters)
2         {
3             filters.Add(new CustomExceptionAttribute(),1);//自定义的验证特性
4             filters.Add(new HandleErrorAttribute(),2);
5         }

 

4.我们在设置了自定义异常处理以后,会跳转到对应的页面,里面的信息相信大家都可以自己定制了

 

5.有图有证据  先来一个404异常处理信息

我请求的页面是home/index,然后自动跳转到了notFound页面
还有500错误,如果您不注意判断一个这个错误,那么你是不可能的

总结一下,在MVC中处理异常有很多中方式,HandleErrorAttribute,自定义异常处理类(重写IExceptionFilter或者是HandleErrorAttribute 的OnException方法),传统的try catch方法,这些都可以,除去那个try catch,其实自定义异常处理就是Filter的体现,和登陆验证没有任何区别。
还有一点就是HttpException的HttpCode为500的情况,ExceptionHandled会被自动设置为true,需要特殊照顾一下。
在我们处理完了异常以后,一定要将其ExceptionHandled设置为true,这样可以避免父类或者更高一级的异常处理捕获处理该异常信息。
MVC处理异常就是这么简单,Filter特性就是那么强大,让我们拥抱MVC,拥抱filter。
我看到有位大牛的MVC 4 书籍已经可以预定了,先试读一下,可以的话买一本,支持国产大牛,丰富一下自己。不是做广告,是发牢骚,什么时候能轮到我出书呢。
 

 

 

posted @ 2012-12-20 23:30  baidixing  阅读(7295)  评论(9编辑  收藏  举报