现在的位置: 首页 > 综合 > 正文

MVC错误处理(一、二)

2012年09月08日 ⁄ 综合 ⁄ 共 6707字 ⁄ 字号 评论关闭

mvc中经常报的错误:

 

“/”应用程序中的服务器错误。


无法找到资源。

说明: HTTP 404。您正在查找的资源(或者它的一个依赖项)可能已被移除,或其名称已更改,或暂时不可用。请检查以下 URL 并确保其拼写正确。 

请求的 URL: /sdf


版本信息: Microsoft .NET Framework 版本:4.0.30319; ASP.NET 版本:4.0.30319.1 

 

--------------------------------------------------------------------------------------------------------------

 

 

MVC中,有一个Filter可以捕捉错误,但是它的用法是利用Attribute来实现的,而且只能加在Controller和Action上,所以不能捕捉别出的错误

其实理论上所有的错误肯定产生于Controller中,但有2种情况下,就不会被捕捉了

1、页面不存在的时候,找不到对应的Controller,那没有任何Controller被执行,所以自然也不会捕捉到错误了

2、在 IAuthorizationFilter 下发生错误的时候,错误捕捉代码在IExceptionFilter中,而IAuthorizationFilter的优先权高于IExceptionFilter,所以也就捕捉不到了

 protected void Application_Error(object sender, EventArgs e)  

复制代码
        {  
            Exception exception = Server.GetLastError();  
            Response.Clear();  
            HttpException httpException 
= exception as HttpException;  
            RouteData routeData 
= new RouteData();  
            routeData.Values.Add(
"controller""Error");  
            
if (httpException == null)  
            {  
                routeData.Values.Add(
"action""Index");  
            }  
            
else //It's an Http Exception, Let's handle it.  
            {  
                
switch (httpException.GetHttpCode())  
                {  
                    
case 404:  
                        
// Page not found.  
                        routeData.Values.Add("action""HttpError404");  
                        
break;  
                    
case 500:  
                        
// Server error.  
                        routeData.Values.Add("action""HttpError500");  
                        
break;  
                    
// Here you can handle Views to other error codes.  
                    
// I choose a General error template    
                    default:  
                        routeData.Values.Add(
"action""General");  
                        
break;  
                }  
            }  
            
// Pass exception details to the target error View.  
            routeData.Values.Add("error", exception.Message);  
            
// Clear the error on server.  
            Server.ClearError();  
            
// Call target Controller and pass the routeData.  
            IController errorController = new WEB.Controllers.ErrorController();  
            errorController.Execute(
new RequestContext(  
           
new HttpContextWrapper(Context), routeData));  
        }  
复制代码

 

 

 

把这段代码放到 Global.asax 中,并且新建一个 Controller 叫做 Error

 

 namespace MVC.Controllers  

复制代码
{  
    public class ErrorController : Controller  
    {  
        
public ActionResult Index(string error)  
        {  
            ViewData[
"Title"= "WebSite 网站内部错误";  
            ViewData[
"Description"= error;  
            
return View("Index");  
        }  
        
public ActionResult HttpError404(string error)  
        {  
            ViewData[
"Title"= "HTTP 404- 无法找到文件";  
            ViewData[
"Description"= error;  
            
return View("Index");  
        }  
        
public ActionResult HttpError500(string error)  
        {  
            ViewData[
"Title"= "HTTP 500 - 内部服务器错误";  
            ViewData[
"Description"= error;  
            
return View("Index");  
        }  
        
public ActionResult General(string error)  
        {  
            ViewData[
"Title"= "HTTP 发生错误";  
            ViewData[
"Description"= error;  
            
return View("Index");  
        }  
    }  
}
复制代码

 

 
这样,就可以捕捉所有错误了。

但其实,这样也不是完美的,因为如果你参考了我第一个问题中,在IIS6下不修改IIS设置,运行了MVC,那当后缀名不是.aspx的时候,错误不会被捕捉

因为这时候输入的地址根本没有交给网站来处理,IIS直接抛出了错误,因为IIS认为这个后缀名不是你所能执行的.

------------------------------------------------------------------------------------------------------------------------------------------------

很多网站如果由于用户输入错了地址,出现了如下图的网页

 

又或者网站的bug导致的应用程序异常,搞出来个满屏的红黄黑,

出现类似情况一定让用户大跌眼镜,个人认为,http错误与应用程序异常的处理方式应该是我们所需关注的问题。

解决方案

1.定义1个枚举类型用来存储http错误码,与应用程序异常错误

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public enum DictSystemErrorType
{
    /// <summary>
    /// 系统错误
    /// </summary>
    SystemError = 1,
    /// <summary>
    /// 系统异常
    /// </summary>
    SystemException = 2,
    /// <summary>
    /// 404错误
    /// </summary>
    Http404Error = 404,
    /// <summary>
    /// 500错误
    /// </summary>
    Http500Error = 500
}
 

2.定义SystemErrorCollection静态类用来管理错误提示信息

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public static class SystemErrorCollection
{
    private static readonly IDictionary<int, string> SystemMsg = new Dictionary<int, string>
                                                           {
                                                               {1,"系统错误,请联系管理员!"},
                                                               {2,"系统出现异常,请联系管理员!"},
                                                               {404,"404错误,Really very sorry,The page not found!"},
                                                               {500,"500错误,Internal Server Error!"},
                                                           };
    /// <summary>
    /// 获取错误提示
    /// </summary>
    /// <param name="errCode"></param>
    /// <returns></returns>
    public static string GetSystemErrorMsg(int  errCode)
    {
        return SystemMsg.SingleOrDefault(p => p.Key == errCode).Value;
    }
}

3.mvc下Global.asax文件和webForm下的一样,都继承自System.Web.HttpApplication,

 他们都包含Application_Error事件(当应用程序中遇到一个未处理的异常时,该事件被触发。

 定义Application_Error事件处理错误与异常

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
protected void Application_Error(object sender, EventArgs e)
{
    Exception exception = Server.GetLastError();
    Response.Clear();
 
    var httpException = exception as HttpException;
    int errorCode = httpException == null ? (int)DictSystemErrorType.SystemException : httpException.GetHttpCode();
 
    //记录log  ...
    //发送错误邮件给网站管理人员  ...
 
    var routeData = new RouteData();
    routeData.Values.Add("controller", "Error");
    routeData.Values.Add("action", "error");
    routeData.Values.Add("errorCode", errorCode);
    Server.ClearError();
 
    IController errorController = new ErrorController();
    errorController.Execute(new RequestContext(new HttpContextWrapper(this.Context), routeData));
}

4.添加ErrorController与Error Action

1
2
3
4
5
6
7
8
9
10
11
12
public class ErrorController : Controller
{
    //
    // GET: /Error/
 
    public ActionResult error()
    {
        int errorCode = (int)(this.RouteData.Values["errorCode"] ?? DictSystemErrorType.SystemError);
        ViewData["errorMsg"] = SystemErrorCollection.GetSystemErrorMsg(errorCode);
        return View();
    }
}

 杂谈

为方便管理错误码与提示信息,定义了枚举类型与一个IDictionary字典。

然而同时维护这两个东西着实有些不变,还好可以通过反射取得枚举的提示信息

 不过最好把错误提示信息对应错误码持久化到数据库或者xml文件中,然后将其缓存起来。

如此可随时更新错误信息,无需修改程序。

 

IController是很简单的,它主要的用途在于提供了关于路由的工具来找到控制器并调用执行(Execute)

Controller的HandleUnknownAction:控制器找不到相关的Action将会呼叫 HandleUnknownAction

另外值得注意的是:

mvc下如果你的某个Controller或者自定义基类的controller重写了HandleUnknownAction方法,

那么出现http404错误的话,该Controller执行完HandleUnknownAction,将不会再执行Application_Error!

-----------------------------------------------------------------------------------------------------------------------------------------------

抱歉!评论已关闭.