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!
-----------------------------------------------------------------------------------------------------------------------------------------------