MVC错误处理(一、二)

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认为这个后缀名不是你所能执行的.

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

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

MVC错误处理(一、二)_第1张图片

 

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

MVC错误处理(一、二)_第2张图片

出现类似情况一定让用户大跌眼镜,个人认为,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!

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

你可能感兴趣的:(mvc)